UIManagedDocument,后台线程和父上下文

时间:2013-05-03 23:53:27

标签: ios core-data uimanageddocument

在UIManagedDocument的文档中简要提到了

  

为了支持异步数据写入,Core Data实际上使用了一对嵌套的托管对象上下文。

特定于UIManagedDocument或Core Data是否总是这样做?

同一份文件还指出

  

如果适用,您可以将后台线程中的数据直接加载到上下文。

这是否意味着在以下代码中

NSManagedObjectContext *moc = self.managedObjectContext;
[moc performBlock:^() {
    Record *record = [NSEntityDescription
        insertNewObjectForEntityForName:@"Record" 
                 inManagedObjectContext:moc];
}];

我应该简单地用

替换第一行
NSManagedObjectContext *moc = self.managedObjectContext.parentContext;

要做到这一点?

我想我也对它的位置感到困惑"合适"直接使用此父上下文。我的意思是听起来像performBlock用于将任务卸载到后台队列。为什么我需要弄乱父上下文?

如果有人为我澄清这一切,我将不胜感激。

1 个答案:

答案 0 :(得分:5)

这是UIManagedDocument的实现细节,但这是核心数据应用程序中的常见设计模式。 managedObjectContext UIManagedDocument是并发类型为NSMainQueueConcurrencyType的上下文,而parentContext是并发类型为NSPrivateQueueConcurrencyType的上下文。

有关并发类型的更多信息,请参阅here。简而言之,父上下文使用后台队列进行操作,而子进程则使用主队列。

通常,当您希望在后台队列中以非阻塞方式执行某些操作时,您希望直接使用父上下文。例如,如果您希望执行长而难以获取的请求,则可以直接在父上下文中执行它。请记住,返回的对象在上下文之间不可互换,因此您必须将返回的对象从一个上下文重新绑定到另一个上下文中(但现在可以使用[NSPredicate predicateWithFormat:@"SELF IN %@", fetchedObjectsFromAnotherContext]轻松执行重新创建)。

总而言之,这取决于你在操作中做了什么。主队列上下文通过不要求您使用performBlock:简化了很多事情,因为所有操作都在主队列(主线程)上执行。如果您只想插入一个对象,移动到专用队列的好处可以忽略不计,或者甚至由于操作系统的上下文切换而有害。但是,如果为了执行此对象插入,您需要执行繁重的工作,可以将其卸载到后台队列并直接在父上下文中执行。