如何将数组保存到CoreData?

140

我需要将我的数组保存到Core Data中。

let array = [8, 17.7, 18, 21, 0, 0, 34]
那个数组中的值和值的数量是可变的。 在我的NSManagedObject类中,我声明什么?
class PBOStatistics: NSManagedObject, Equatable {
    @NSManaged var date: NSDate
    @NSManaged var average: NSNumber
    @NSManaged var historicAverage: NSNumber
    @NSManaged var total: NSNumber
    @NSManaged var historicTotal: NSNumber
    @NSManaged var ordersCount: NSNumber
    @NSManaged var historicOrdersCount: NSNumber
    @NSManaged var values: [Double]  //is it ok?

    @NSManaged var location: PBOLocation

}

2. 我应该在我的.xcdatamodel文件中声明什么?

在图片中输入图像描述

3. 我如何将这个保存到我的实体中?(我使用 MagicalRecord)

let statistics = (PBOStatistics.MR_createInContext(context) as! PBOStatistics)
statistics.values = [8, 17.7, 18, 21, 0, 0, 34] //is it enough?

没有"应该这样做",数据库设计由你决定。例如,在我看来,如果在你的应用程序中存储这些数据最有效的方式是使用日期或文本格式,那么你完全可以使用它们。 - A-Live
在我的NSManagedObject中,@NSManaged var values: [Double]是可以的吗?你能告诉我在.xcdatamodel中应该使用什么类型来保存它吗? - Bartłomiej Semańczyk
我可能没有表达清楚,你的提问涉及个人偏好,没有真正需要解决的问题。如果你正在寻找使用一对多关系的方法,请添加你尝试过什么以及遇到了什么问题的信息。如果你理解了你提到的每种解决方案,并且正在寻找最有效的解决方案-列出你的效率标准并描述使用情况。如果由于某些原因你难以理解不同类型的关系或根本不想使用关系-直接说出来。 - A-Live
我更新了问题。 - Bartłomiej Semańczyk
这是一个不错的问题,我已经为您添加了MagicalRecord标签,但不幸的是,我在这个领域没有经验,希望有经验的人能够从这一点更好地帮助您。 - A-Live
6个回答

223

好的,我进行了一些研究和测试。使用Transformable类型,解决方案很简单:

1. 我在我的NSManagedObject类中声明什么?

@NSManaged var values: [NSNumber]  //[Double] also works

2. 我需要在我的.xcdatamodel文件中声明什么?

Transformable 数据类型。

3. 我该如何将其保存在实体中?

statistics!.values = [23, 45, 567.8, 123, 0, 0] //just this

“您可以将NSArray或NSDictionary存储为可转换属性。这将使用NSCoding将数组或字典序列化为NSData属性(并在访问时适当反序列化)”- 来源

或者如果您想将其声明为二进制数据,那么请阅读这篇简单的文章


4
也适用于 [NSString] 表示字符串数组。 - Daisy R.
在从Core Data获取数据时,是否可以添加谓词? - Satyam
5
我正在尝试使用一个字符串数组,但它无法正常工作。 - allemattio
你会用 Swift 3 怎么做这个? - MartianMartian
2
看起来这个在未来的Swift版本中将不再适用:https://dev59.com/1FIG5IYBdhLWcg3w21eA - FontFamily
1
对于字符串数组,将数据类型设置为Transformable,并在特定属性的属性面板中将其转换器类型设置为“NSSecureUnarchiveFromDataTransformer”。 - AfnanAhmad

146

Swift 3 由于 Swift 3 不再具有实现文件,因此我们需要进入 xcdatamodeld 文件并选择实体和所需属性(在这个示例中它被称为 values)。 将其设置为可转换的,并将其自定义类设置为 [Double]。 现在可以像使用普通数组一样使用它。

将自定义类设置为 Double 数组


能否将自定义类定义为 [[String: Any]]?一个字典数组。 - BergP
理论上来说应该是可以接受的,不过我从来没试过。 - Hola Soy Edu Feliz Navidad
3
你可以将codegen设置为Manual/None并编写自己的Core Data模型类。 - Schemetrical
好方法,TIL - Stanislau Baranouski
11
另外,将NSSecureUnarchiveFromData设置为Value Transformer(它已被重命名为“Transformer”)。否则,您可能会收到此运行时警告:“NSKeyedUnarchiveFromData”不应用于取消归档,并将在未来的版本中移除 - Manabu Nakazawa
关于Int数组,这将如何工作?属性类型有Int16、Int32、Int64。 - YHStan

16

将数组转换为NSData

let appDelegate =
    UIApplication.sharedApplication().delegate as! AppDelegate
let managedContext = appDelegate.managedObjectContext
let entity =  NSEntityDescription.entityForName("Device",
                                                inManagedObjectContext:managedContext)
let device = NSManagedObject(entity: entity!,
                             insertIntoManagedObjectContext: managedContext)
let data = NSKeyedArchiver.archivedDataWithRootObject(Array)

device.setValue(data, forKey: "dataOfArray")
do {
    try managedContext.save()
    devices.append(device)
} catch let error as NSError  {
    print("Could not save \(error), \(error.userInfo)")
}

选择二进制数据

将 NSData 转换为数组

let appDelegate =
    UIApplication.sharedApplication().delegate as! AppDelegate
let managedContext = appDelegate.managedObjectContext
let fetchRequest = NSFetchRequest(entityName: "Device")

do {
    let results =
        try managedContext.executeFetchRequest(fetchRequest)

    if results.count != 0 {

        for result in results {

                let data = result.valueForKey("dataOfArray") as! NSData
                let unarchiveObject = NSKeyedUnarchiver.unarchiveObjectWithData(data)
                let arrayObject = unarchiveObject as AnyObject! as! [[String: String]]
                Array = arrayObject
        }

    }

} catch let error as NSError {
    print("Could not fetch \(error), \(error.userInfo)")
}

例如:https://github.com/kkvinokk/Event-Tracker


1
使用NSData确实更合理和更简单。 - GeneCode
2
如果您想将谓词添加到该属性中,可能无法正常工作。 - Satyam

13

如果希望简单地将数组存储为字符串

尝试这样做:

// Array of Strings
let array: [String] = ["red", "green", "blue"]
let arrayAsString: String = array.description
let stringAsData = arrayAsString.data(using: String.Encoding.utf16)
let arrayBack: [String] = try! JSONDecoder().decode([String].self, from: stringAsData!)

对于其他数据类型:

// Set of Doubles
let set: Set<Double> = [1, 2.0, 3]
let setAsString: String = set.description
let setStringAsData = setAsString.data(using: String.Encoding.utf16)
let setBack: Set<Double> = try! JSONDecoder().decode(Set<Double>.self, from: setStringAsData!)

1
你的解决方案节省了我很多时间。 - Kishore Kumar
我不确定为什么,但这些解决方案都对我无效。我尝试了您的解决方案,将字符串数组的数据存储在Core Data中并检索它,然后解码为字符串数组,但仍然无法正常工作。有什么建议吗? - Tyler Cheek
1
这只是一种相当粗糙的方法——数据不会被索引在数据库中,因为它被存储为BLOB。读/写数据也有相当多的开销——例如,您不能使用任何集合的方法轻松地进行更改。使用transformable是首选方式。 - vsanthanam510
1
不错!我还会添加以下可能性,这在某些情况下可能有效!var array = ["red", "green", "blue"] var arrayString = array.joined(separator: "%20") var arrayBack = arrayString.components(separatedBy: "%20") - Laura Corssac

4
将实体属性类型设置为“二进制数据”。
NSData *arrayData = [NSKeyedArchiver archivedDataWithRootObject:TheArray];
myEntity.arrayProperty = arrayData;
[self saveContext]; //Self if we are in the model class

将原始数组检索为:

NSMutableArray *array = [NSKeyedUnarchiver unarchiveObjectWithData:anEntity.arrayProperty];

这就是全部。


2

以下代码适用于我将JSON数组存储在CoreData中

func saveLocation(model: [HomeModel],id: String){

    let newUser = NSEntityDescription.insertNewObject(forEntityName: "HomeLocationModel", into: context)

    do{
        var dictArray = [[String: Any]]()
        for i in 0..<model.count{
            let dict = model[i].dictionaryRepresentation()
            dictArray.append(dict)
        }
        let data = NSKeyedArchiver.archivedData(withRootObject: dictArray)
        newUser.setValue(data, forKey: "locations")
        newUser.setValue(id, forKey: "id")
        try context.save()
    }catch {
       print("failure")
    }

}

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接