在CoreData中保存Swift CLLocation

时间:2014-11-06 22:20:49

标签: core-data swift cllocation nsvalue

我正在尝试将CLLocation数据保存在Core Data中。我的属性设置为可转换对象。

那里的一些示例代码表示可以使用转换为NSValue的objc保存位置数据。

CLLocationCoordinate2D myLocation;
NSValue *locationValue = [NSValue valueWithBytes:&myLocation objCType:@encode(CLLocationCoordinate2D)]
// then put locationValue into storage - maybe an ObjC collection class or core data etc.

从数据存储中检索对象应该可以使用这个范例。

CLLocationCoordinate2D myLocation;
NSValue *locationValue = // restored from your database or retrieved from your collection class
[locationValue getValue:&myLocation];

我试图推断这个想法并将整个CLLocation对象捕获到存储在NSValue中的CoreData,其目标是从CLLocation恢复CLLocation对象以后存储。

在测试各种方法时,我发现在尝试从Core Data Store中解包CLLocation对象时返回了nil。我使用以下内容将//verified that I have a good CLLocation here newManagedObject.setValue(location as CLLocation, forKey: "location") 对象放入商店:

CLLocation

稍后我尝试打开let location = detail.valueForKey("location")! as CLLocation

(CLLocation) location = 0x00007fff59537c90 {
  ObjectiveC.NSObject = {}
}

但我发现该物体没有正确打开。

cllocation

展开的对象看起来像:

CLLocation

因此,此时对CLLocation对象的任何使用都会返回" 找到的nil ...错误"

我是否遗漏了使用Swift存储/检索//assuming location is a valid CLLocation and data model has a binary data field let archivedLocation = NSKeyedArchiver.archivedDataWithRootObject(location) newManagedObject.setValue(archivedLocation, forKey: "location") 个对象的明显内容?

更新

丹尼尔的回答让我意识到NSValueTransformer不适合我,所以我回到了以前用过的方法。基本上使用NSKeyedArchiver编码对象图。我的目标一如既往地使用最简单的解决方案,因此我选择不使用托管对象子类,因为我需要的属性已经存在于现有的CLLocation对象中。我选中了选项(存储在外部记录文件中),将可转换字段更改为二进制数据。我开始测试这个过程,只需对主细节模板进行最小的更改。

data-model

保存CLLocation对象图

let object = self.fetchedResultsController.objectAtIndexPath(indexPath) as NSManagedObject
let controller = (segue.destinationViewController as UINavigationController).topViewController as DetailViewController
                    controller.detailItem = object 

从数据存储中恢复对象图

在主vc上作为默认的一部分准备segue,数据通过fetch控制器恢复。

var lc = NSKeyedUnarchiver.unarchiveObjectWithData(detail.valueForKey!("location") as NSData) as CLLocation

在详细信息vc上,我使用NSKeyedUnarchiver

打开我的对象图
{{1}}

3 个答案:

答案 0 :(得分:5)

NSManagedObject对象创建CLLocation子类时,您将遇到的麻烦要少得多。然后创建用于存储和检索的方法以方便:

import Foundation
import CoreData
import CoreLocation

class LocationPoint: NSManagedObject {

    @NSManaged var latitude: NSNumber!
    @NSManaged var longitude: NSNumber!
    @NSManaged var altitude: NSNumber!
    @NSManaged var timestamp: NSDate!
    @NSManaged var horizontalAccuracy: NSNumber
    @NSManaged var verticalAccuracy: NSNumber
    @NSManaged var speed: NSNumber
    @NSManaged var course: NSNumber

    func initFromLocation(location: CLLocation) {
        self.latitude           = location.coordinate.latitude
        self.longitude          = location.coordinate.longitude
        self.altitude           = location.altitude
        self.timestamp          = location.timestamp

        self.horizontalAccuracy = location.horizontalAccuracy > 0.0 ? location.horizontalAccuracy : 0.0
        self.verticalAccuracy   = location.verticalAccuracy > 0.0 ? location.verticalAccuracy : 0.0
        self.speed              = location.speed > 0.0 ? location.speed : 0.0
        self.course             = location.course > 0.0 ? location.course : 0.0
    }

    func location() -> CLLocation {
        return CLLocation(
            coordinate: CLLocationCoordinate2D(latitude: self.latitude.doubleValue, longitude: self.longitude.doubleValue),
            altitude: self.altitude.doubleValue,
            horizontalAccuracy: self.horizontalAccuracy.doubleValue,
            verticalAccuracy: self.verticalAccuracy.doubleValue,
            course: self.course.doubleValue,
            speed: self.speed.doubleValue,
            timestamp: self.timestamp
        )
    }

您必须根据此类设置实体数据模型。

答案 1 :(得分:3)

你缺少的是NSValue用于存储类型,即连续的内存块,而不是对象。 CLLocation是一个对象。

为了在核心数据中存储CLLocation,您必须存储每个属性。另一种方法是使用编码器。

答案 2 :(得分:1)

这篇文章仍然有帮助,但有点过时了。

要保存CLLocation,请使用:

let clocation: [CLLocation]

let archivedLocation = NSKeyedArchiver.archivedData(withRootObject: clocation)

并获取核心数据:

let unarchivedLocation = NSKeyedUnarchiver.unarchiveObject(with: (coreDataLocation.value(forKey: "location") as! NSData) as Data) as! CLLocation)