在 CoreData 中保存 Swift CLLocation

Posted

技术标签:

【中文标题】在 CoreData 中保存 Swift CLLocation【英文标题】:Saving Swift CLLocation in CoreData 【发布时间】:2014-11-06 22:20:48 【问题描述】:

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

那里的一些示例代码表明可以使用转换为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 对象捕获到存储在CoreData 中的NSValue 中,目标是稍后从存储中恢复CLLocation 对象。

在测试各种方法时,我发现在尝试从 Core Data Store 解包 CLLocation 对象时返回了 nil。我正在使用以下方法将CLLocation 对象放入商店:

//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 对象的任何使用都会返回“found nil...error

我是否遗漏了一些关于使用 Swift 存储/检索 CLLocation 对象的明显内容?

更新

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

保存 CLLocation 对象图

//assuming location is a valid CLLocation and data model has a binary data field
let archivedLocation = NSKeyedArchiver.archivedDataWithRootObject(location)
newManagedObject.setValue(archivedLocation, forKey: "location")

从数据存储中恢复对象图

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

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

在细节 vc 上,我使用 NSKeyedUnarchiver 解开我的对象图

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

【问题讨论】:

在“保存 CLLocation 对象图”下,什么是 newManagedObject?你能显示你启动它的代码行吗?谢谢。 【参考方案1】:

当您为CLLocation 对象创建NSManagedObject 子类时,麻烦会少得多。然后为方便起见创建存储和检索方法:

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
        )
    

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

【讨论】:

感谢 zisoft,这个解决方案没有问题,但我更喜欢让事情尽可能简单。我尽可能避免使用 NSManaged Object 子类。我最终使用 NSKeyedArchiver 根据问题中的更新捕获 CLLocation 的对象图。【参考方案2】:

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

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

【讨论】:

谢谢,这非常有用。我决定将整个对象归档到一个数据字段中。所以用 NSValueTransformer 结束实验:)【参考方案3】:

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

为了保存我使用的 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)

【讨论】:

以上是关于在 CoreData 中保存 Swift CLLocation的主要内容,如果未能解决你的问题,请参考以下文章

将 CoreData 保存到 iCloud (Swift)

如何将 UIStepper 保存到 CoreData (swift 3)

在 Swift 中将 CoreData 保存/同步到 iCloud

保存/获取 CoreData 关系 Swift 3

如何在 sqlite 或 CoreData 中保存位置坐标,并在 swift 中进行重要的位置监控

如何在 Swift 中将位置面包屑保存到 CoreData