iOS 9中实例化NSManagedObjectContext的正确姿势 - Core Data

简书上的文章已经不再维护,有兴趣阅读其他文章,或一起交流的朋友,请移步 我的博客:punmy.cn

原文


这两天在使用Core Data的过程中遇到了一个问题,在执行到managedObjectContext 的 -performBlock: 方法的时候,应用崩溃并提示:

  • Can only use -performBlock: on an NSManagedObjectContext that was created with a queue

为何用 -performBlock:

我们之所以使用 performBlock: 方法,而不是直接进行 Core Data 的相关操作,是因为 Core Data 的操作不是线程安全的
使用 -performBlock: 方法可以确保 Core Data 的相关操作都在 managedObjectContext 所在的线程上完成。

崩溃原因

崩溃的错误信息比较清楚的表明:“只能有和 queue 一起创建的NSManagedObjectContext 实例才能使用 -performBlock: 方法”。
这说明我们在创建 NSManagedObjectContext 实例的时候,没有为它“分配”一个 queue。
下面我就为大家介绍一下实例化 NSManagedObjectContext 的正确姿势。

正确姿势

我们知道, iOS 9 开始,苹果就在使用 Core Data 的项目中的 APPDelegate.m 中预先为我们提供了一份配置 Core Data Stack 的模板代码。
这份代码中就有默认的实例化 NSManagedObjectContext 的代码。

- (NSManagedObjectContext *)managedObjectContext {
    // Returns the managed object context for the application (which is already bound to the persistent store coordinator for the application.)
    if (_managedObjectContext != nil) {
        return _managedObjectContext;
    }
    
    NSPersistentStoreCoordinator *coordinator = [self persistentStoreCoordinator];
    if (!coordinator) {
        return nil;
    }
    _managedObjectContext = [[NSManagedObjectContext alloc] init]; //默认的实例化
    [_managedObjectContext setPersistentStoreCoordinator:coordinator];
    return _managedObjectContext;
}

这个默认的实例化代码使用了 init 方法,这并不合适,因为这样就不能为 NSManagedObjectContext 实例提供一个线程。我们需要将这个 init 方法替换成 -initWithConcurrency: 方法。这个方法配置了 NSManagedObjectContext 实例化所在的线程。
这就意味着我们要确定在哪个线程上实例化我们的 NSManagedObjectContext ,主线程,还是另外创建一个后台线程。我们可以选择的参数有:

  • NSPrivateQueueConcurrencyType
  • NSMainQueueConcurrencyType

在这里,我把它配置成在主线程上进行实例化(一般选择主线程就可以)。代码如下:

- (NSManagedObjectContext *)managedObjectContext {
    // Returns the managed object context for the application (which is already bound to the persistent store coordinator for the application.)
    if (_managedObjectContext != nil) {
        return _managedObjectContext;
    }
    
    NSPersistentStoreCoordinator *coordinator = [self persistentStoreCoordinator];
    if (!coordinator) {
        return nil;
    }

    // since iOS 9
    _managedObjectContext = [[NSManagedObjectContext alloc] initWithConcurrencyType:NSMainQueueConcurrencyType];
    [_managedObjectContext setPersistentStoreCoordinator:coordinator];
    return _managedObjectContext;
}

这样问题就解决啦。

更详细的内容可以参考Apple 官方文档:
https://developer.apple.com/library/prerelease/ios/documentation/Cocoa/Reference/CoreDataFramework/Classes/NSManagedObjectContext_Class/#//apple_ref/c/tdef/NSManagedObjectContextConcurrencyType

推荐阅读更多精彩内容

  • 本文是对 MagicalRecord github上的翻译 正文:注意: MagicalRecord 在 ARC...
    騂跃神话阅读 1,538评论 1 5
  • 1.Difference between shallow copy and deep copy? 浅复制和深复制的...
    用心在飞阅读 835评论 0 9
  • 之前看了很多面试题,感觉要不是不够就是过于冗余,于是我将网上的一些面试题进行了删减和重排,现在分享给大家。(题...
    Job_Yang阅读 11,888评论 12 144
  • __block和__weak修饰符的区别其实是挺明显的:1.__block不管是ARC还是MRC模式下都可以使用,...
    LZM轮回阅读 2,837评论 0 6
  • Object-c的类可以多重继承么?可以实现多个接口么?Category是什么?重写一个类的方式用继承好还是分类好...
    small_Sun阅读 608评论 0 0
  • 付付网名兰心,取蕙质兰心之意,人如其名,付付就如那朵在幽谷中独自摇曳的蕙兰,于清风中散发馥郁芬芳,却不染...
    蔽日幽竹阅读 520评论 0 1
  • 今天,第一次将自己写的文档在简书上分享,觉得蛮好的。以后,自己会主要做有关分布式对象存储的事情,尤其是gluste...
    zhllsr阅读 182评论 0 0