使用NSPersistentContainer进行核心数据并发

时间:2017-07-18 20:51:36

标签: ios objective-c core-data concurrency

注意:我看过类似的问题,但没有找到描述这种情况的问题。

我正在查看Apple关于Core Data并发性的以下示例代码(https://developer.apple.com/library/content/documentation/Cocoa/Conceptual/CoreData/Concurrency.html

NSArray *jsonArray = …;
NSPersistentContainer *container = self.persistentContainer;
[container performBackgroundTask:^(NSManagedObjectContext *context) {
    for (NSDictionary *jsonObject in jsonArray) {
        AAAEmployeeMO *mo = [[AAAEmployeeMO alloc] initWithContext:context];
        [mo populateFromJSON:jsonObject];
    }
    NSError *error = nil;
    if (![context save:&error]) {
        NSLog(@"Failure to save context: %@\n%@", [error localizedDescription], [error userInfo]);
        abort();
    }
}];

在我的应用程序中,直到用户点击屏幕上的保存按钮才会启动保存。如何使用子上下文代替那种私有上下文是VC的属性的情境?

NSArray *jsonArray = …; //JSON data to be imported into Core Data
NSManagedObjectContext *moc = self.persistentContainer.viewContext; //Our primary context on the main queue

NSManagedObjectContext *private = [[NSManagedObjectContext alloc] initWithConcurrencyType:NSPrivateQueueConcurrencyType];
[private setParentContext:moc];

[private performBlock:^{
    for (NSDictionary *jsonObject in jsonArray) {
        NSManagedObject *mo = …; // WHICH CONTEXT TO USE?  <<<======
        //update MO with data from the dictionary
    }
    NSError *error = nil;
    if (![private save:&error]) {
        NSLog(@"Error saving context: %@\n%@", [error localizedDescription], [error userInfo]);
        abort();
    }
}

然后,一旦用户点击保存,请执行以下操作:

NSManagedObjectContext *moc = self.persistentContainer.viewContext; //Our primary context on the main queue
   [moc performBlockAndWait:^{
            NSError *error = nil;
            if (![moc save:&error]) {
                NSLog(@"Error saving context: %@\n%@", [error localizedDescription], [error userInfo]);
                abort();
            }
        }];
    }];

还要注意上面示例中使用哪个moc的问题(&lt;&lt;&lt;&lt; =====)

编辑:我最后做的是立即保存子上下文,以便表只使用viewContext来显示结果。如果用户退出而未保存,我会再次从viewContext删除所有结果。保存按钮仍然存在,但现在只设置一个标志,表示不删除结果。

1 个答案:

答案 0 :(得分:1)

如果您有一页表单,并且希望在用户按下保存按钮时保存表单,则只需从textFields(或您输入的数据)中获取数据,然后使用{{1}将其放入核心数据中}}。由于数据仅在用户正在编辑时存储在textField中,如果用户推回他的编辑将丢失。

如果您对具有许多不同实体的复杂文档进行了大量更改,用户可以创建或销毁或链接这些更改,并且只有在用户按下save时才会保存所有这些更改,那么您应该使用子上下文。您将根据子上下文中的值显示数据,但只有在用户按下save时才将这些更改推送到父上下文。这是一种非常罕见的情况,我从未亲自遇到过这样做的必要性。

我强烈怀疑你是第一种情况。不要使用子上下文。使用performBackgroundTask并在用户按保存时保存数据。

(在performBackgroundTask上下文中使用的正确上下文是[private performBlock:^{上下文)