Jak zapisać Array w CoreData?

93

Muszę zapisać moją tablicę w danych podstawowych.

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

Wartości wewnątrz tej tablicy i liczba wartości są zmienne.

1. Co deklaruję w mojej klasie 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. Co mam zadeklarować w moim .xcdatamodel?

wprowadź opis obrazu tutaj

3. Jak mogę to zapisać w mojej jednostce? (Używam MagicalRecord)

let statistics = (PBOStatistics.MR_createInContext(context) as! PBOStatistics)
statistics.values = [8, 17.7, 18, 21, 0, 0, 34] //is it enough?
Bartłomiej Semańczyk
źródło
Nie ma „powinno”, projekt bazy danych zależy od Ciebie, np. Jeśli o mnie chodzi, równie dobrze możesz użyć dat lub formatu tekstowego, jeśli okaże się to najbardziej efektywnym sposobem przechowywania tych danych w Twojej aplikacji.
A-Live
Więc w moim NSManagedObject: @NSManaged var values: [Double]czy to dobrze? Czy możesz mi powiedzieć, jakiego typu mam użyć, .xcdatamodelaby to zapisać?
Bartłomiej Semańczyk
Być może nie wyjaśniłem tego wystarczająco jasno, sposób, w jaki pytasz, dotyczy osobistych preferencji, nie ma prawdziwego problemu do rozwiązania. Jeśli szukasz sposobu na użycie relacji jeden do wielu, dodaj informacje o tym, co próbowałeś i gdzie wystąpił problem. Jeśli rozumiesz każde z wymienionych przez Ciebie rozwiązań i szukasz najskuteczniejszego - wypisz swoje kryteria wydajności i opisz przypadki użycia. Jeśli z jakiegoś powodu masz kłopoty ze zrozumieniem różnego rodzaju relacji lub w ogóle nie chcesz ich używać - powiedz to wprost.
A-Live
Zaktualizowałem pytanie
Bartłomiej Semańczyk
To przyzwoite pytanie, dodałem dla ciebie tag MagicalRecord, niestety nie mam doświadczenia w tej dziedzinie i mam nadzieję, że ktoś, kto będzie w stanie pomóc ci lepiej od tego momentu.
A-Live

Odpowiedzi:

179

Ok, przeprowadziłem badania i testy. Korzystanie Transformable typ, rozwiązanie jest proste:

1. Co deklaruję w mojej klasie NSManagedObject?

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

2. Co mam zadeklarować w moim .xcdatamodel?

Transformable typ danych.

3. Jak mogę to zapisać w mojej jednostce?

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

„Można przechowywać NSArray lub NSDictionary jako atrybut podlegający transformacji. Spowoduje to użycie kodu NSCoding do serializacji tablicy lub słownika do atrybutu NSData (i odpowiedniej deserializacji po uzyskaniu dostępu) ”- źródło

Lub jeśli chcesz zadeklarować je jako dane binarne, przeczytaj ten prosty artykuł :

Bartłomiej Semańczyk
źródło
Dobra robota, odpowiedź wydaje się warta zielonego znacznika wyboru ze względu na ukończenie.
A-Live
3
Działa również z [NSString]tablicą smyczkową
Daisy R.
Czy można dodać predykat podczas pobierania z podstawowych danych?
Satyam
5
W moim przypadku Próbuję użyć tablicę łańcuchów i to nie działa
allemattio
jak byś to zrobił w szybki 3 sposób?
Martian2049
97

Swift 3 Ponieważ od wersji Swift 3 nie mamy już plików implementacji, musimy przejść do pliku xcdatamodeld, wybrać jednostkę i żądany atrybut (w tym przykładzie nazywa się to wartościami). Ustaw go jako transformowalny, a jego klasę niestandardową na [Double]. Teraz użyj go jako zwykłej tablicy.

Ustawienie klasy niestandardowej na tablicę Double

Hola Soy Edu Feliz Navidad
źródło
Czy można zdefiniować klasę niestandardową jako [[String: Any]]? Szereg słowników
BergP
Teoretycznie powinno być do zaakceptowania, ale nigdy nie próbowałem.
Hola Soy Edu Feliz Navidad
2
Zawsze możesz ustawić codegen na Manual / None i napisać własną klasę modelu Core Data.
Schematyczny
14

Konwertuj Array na 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)")
}

Wybierz dane binarne

Konwertuj NSData na Array

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)")
}

Na przykład: https://github.com/kkvinokk/Event-Tracker

Vinoth Anandan
źródło
1
Używanie NSData rzeczywiście ma większy sens i jest prostsze.
GeneCode
2
Jeśli chcesz dodać predykat do tego atrybutu, może to nie działać.
Satyam
6

Jeśli zachowasz prostotę i zapisz tablicę jako ciąg

Spróbuj tego:

// 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!)

W przypadku innych typów danych odpowiednio:

// 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!)
Alexey Chekanov
źródło
Twoje rozwiązanie zaoszczędziło mi dużo czasu
Kishore Kumar
Nie wiem dlaczego, ale żadne z tych rozwiązań nie działa. Wypróbowałem rozwiązanie polegające na przechowywaniu danych tablicy ciągów w danych podstawowych i pobieraniu ich, a następnie dekodowaniu na tablicę ciągów, ale nadal nie działa. Jakakolwiek rada?
Tyler Cheek
3

Ustaw typ atrybutu encji na „Dane binarne”

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

Przywróć oryginalną macierz jako:

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

To wszystko.

Avijit Nagare
źródło
1

Poniższy kod działa dla mnie do przechowywania tablicy JSON w 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")
    }

}
Rahul Gusain
źródło