Как сохранить массив в CoreData?

93

Мне нужно сохранить свой массив в Core Data.

let array = [8, 17.7, 18, 21, 0, 0, 34]

Значения внутри этого массива и количество значений являются переменными.

1. Что я объявляю внутри моего класса 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. Как мне сохранить это в моем Entity? (Я использую 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чтобы сохранить это?
Бартломей Семанчик
Возможно, я недостаточно ясно дал понять, что вы спрашиваете о личных предпочтениях, реальной проблемы, которую нужно решать, нет. Если вы ищете способ использовать отношения «один ко многим», добавьте информацию о том, что вы пробовали и где у вас возникла проблема. Если вы понимаете каждое из упомянутых вами решений и ищете наиболее эффективное - перечислите свои критерии эффективности и опишите варианты использования. Если по какой-то причине у вас возникли проблемы с пониманием различных видов отношений или вы вообще не хотите использовать отношения - скажите прямо.
A-Live
Я обновил вопрос
Bartłomiej Semańczyk
Это достойный вопрос, я добавил для вас тег MagicalRecord, к сожалению, у меня нет опыта в этой области, и, надеюсь, кто-то, кто сможет помочь вам лучше с этого момента.
A-Live

Ответы:

179

Хорошо, я провел небольшое исследование и тестирование. Используя трансформируемый тип, решение просто:

1. Что я объявляю внутри моего класса NSManagedObject?

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

2. Что я объявляю в моей .xcdatamodel?

Transformable тип данных.

3. Как мне сохранить это в моем Entity?

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

«Вы можете хранить NSArray или NSDictionary как трансформируемый атрибут. Это будет использовать NSCoding для сериализации массива или словаря в атрибут NSData (и соответствующей десериализации при доступе) »- Источник

Или, если вы хотите объявить его как двоичные данные, прочтите эту простую статью :

Бартломей Семанчик
источник
Хорошая работа, ответ, похоже, заслуживает зеленой галочки для завершения.
A-Live
3
Также работает с [NSString]массивом строк
Дейзи Р.
Можно ли добавить предикат при выборке из основных данных?
Satyam
5
В моем случае я пытаюсь использовать массив строк, и это не работает
allemattio
как бы вы сделали это быстрым 3 способом?
Martian2049
97

Swift 3 Поскольку в Swift 3 у нас больше нет файлов реализации, нам нужно перейти к файлу xcdatamodeld, выбрать сущность и желаемый атрибут (в этом примере он называется значениями). Установите его как трансформируемый, а его собственный класс - [Double]. Теперь используйте его как обычный массив.

Установка пользовательского класса в массив Double

Хола Сой Эду Фелиз Навидад
источник
Можно ли определить собственный класс как [[String: Any]]? Множество словарей
BergP
Теоретически это должно быть приемлемо, но я никогда не пробовал.
Hola Soy Edu Feliz Navidad
2
Вы всегда можете установить codegen в значение Manual / None и написать свой собственный класс модели Core Data.
Схема
14

Преобразовать массив в 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
6

Если сохранить простоту и сохранить массив как строку

Попробуй это:

// 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!)
Алексей Чеканов
источник
Ваше решение сэкономило мне много времени
Кишор Кумар
Я не уверен, почему, но ни одно из этих решений не работает для меня. Я попробовал ваше решение по хранению данных строкового массива в Core Data и их извлечению, а затем декодированию их в строковый массив, но оно все еще не работает. Любой совет?
Тайлер Чик
3

Сделайте тип атрибута объекта как "Двоичные данные"

NSData *arrayData = [NSKeyedArchiver archivedDataWithRootObject:TheArray];
myEntity.arrayProperty = arrayData;
[self saveContext]; //Self if we are in the model class

Получить исходный массив как:

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

Вот и все.

Авиджит Нагаре
источник
1

Следующий код работает для меня для хранения массива 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")
    }

}
Рахул Гусейн
источник