Как добавить NSManagedObjectContext с ConcerrencyType в swift 3?

#ios #core-data #swift3

#iOS #core-данные #swift3

Вопрос:

В моем приложении у меня есть задачи, которые загружают данные и сохраняют их в CoreData. После того, как я перешел на swift 3, он начал генерировать исключения при сохранении данных в случайное время. Как я понял, это может произойти, если я использую один контекст для всех операций. Теперь я создал другой контекст с типом параллелизма, все работает без ошибок, но в файл .sqlite ничего не сохраняется 🙂 Вот как я создаю контекст:

 static let context : NSManagedObjectContext = (UIApplication.shared.delegate as! AppDelegate).persistentContainer.viewContext
static let privateContext = NSManagedObjectContext(concurrencyType: NSManagedObjectContextConcurrencyType.privateQueueConcurrencyType)

static func declare()
{
    AixmParser.privateContext.parent = AixmParser.context
}
  

Я сохраняю данные следующим образом:

     do{
        try privateContext.save()
    }
    catch{
    }
  

Нужно ли что-то добавлять в объявление или в основной стек данных?

Обновление: основной стек данных ниже.

 // MARK: - Core Data stack

lazy var persistentContainer: NSPersistentContainer = {
    /*
     The persistent container for the application. This implementation
     creates and returns a container, having loaded the store for the
     application to it. This property is optional since there are legitimate
     error conditions that could cause the creation of the store to fail.
    */
    let container = NSPersistentContainer(name: "AppName")
    container.loadPersistentStores(completionHandler: { (storeDescription, error) in
        if let error = error as NSError? {
            // Replace this implementation with code to handle the error appropriately.
            // fatalError() causes the application to generate a crash log and terminate. You should not use this function in a shipping application, although it may be useful during development.

            /*
             Typical reasons for an error here include:
             * The parent directory does not exist, cannot be created, or disallows writing.
             * The persistent store is not accessible, due to permissions or data protection when the device is locked.
             * The device is out of space.
             * The store could not be migrated to the current model version.
             Check the error message to determine what the actual problem was.
             */
            fatalError("Unresolved error (error), (error.userInfo)")
        }
        container.viewContext.perform({
            // actions upon the NSMainQueueConcurrencyType NSManagedObjectContext for this container
        })

    })
    return container
}()

// MARK: - Core Data Saving support

func saveContext () {
    let context = persistentContainer.viewContext
    if context.hasChanges {
        do {
            try context.save()
        } catch {
            // Replace this implementation with code to handle the error appropriately.
            // fatalError() causes the application to generate a crash log and terminate. You should not use this function in a shipping application, although it may be useful during development.
            let nserror = error as NSError
            fatalError("Unresolved error (nserror), (nserror.userInfo)")
        }
    }
}
  

Комментарии:

1. Нужно посмотреть, как вы настроили свой основной стек данных. save() Сохраняет значения и отправляет их в parentContext . Вы должны связать save() в while цикле с последним контекстом без родительского элемента. Этот контекст напрямую связан с persistentStoreCoordinator .

2. @New16, я добавил свой основной стек данных. Как я должен связать save() ?

Ответ №1:

Итак, во-первых, вам нужно создать ссылку из privateContext NSPersistentContainer «s viewContext » . Итак, при создании частного, privateContext , privateContext.parentContext = viewContext . При сохранении let context = privateContext
while(context != nil) {
context.save()
context = context.privateContext
}
это будет работать. Я не уверен, почему вы получили ошибки в первую очередь.

Комментарии:

1. Разве это не то же самое? Я создаю контекст, который является viewContext, а затем устанавливаю его как parentContext для privateContext. О сохранении — я должен добавить его в основной стек данных или просто заменить свой код сохранения везде?

2. Создайте extension of NSManagedObjectContext , чтобы включить распространение save() метода. Итак, вместо вызова save() call like saveAndPropagate() .

3. context = context.privateContext Вы имели в виду context.parent здесь?

4. Хорошо. extension NSManagedObject { func privateContext() -> self { let context = NSManagedObjectContext(concurrencyType:.privateQueueConcurrencyType) context.parentContext = (UIApplication.shared.delegate as! AppDelegate).persistentContainer.viewContext return context} func saveAndPropagate() { let context = privateContext while(context != nil) { context.save() context = context.privateContext }} }

5. я прочитал некоторую документацию и сделал почти то же самое, отправил все запросы к CoreData в singleton и добавил функцию для создания контекста для фона. Спасибо!