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...
    騂跃神话阅读 2,080评论 1 5
  • 1.Difference between shallow copy and deep copy? 浅复制和深复制的...
    用心在飞阅读 1,044评论 0 9
  • 之前看了很多面试题,感觉要不是不够就是过于冗余,于是我将网上的一些面试题进行了删减和重排,现在分享给大家。(题...
    Job_Yang阅读 12,132评论 12 143
  • __block和__weak修饰符的区别其实是挺明显的:1.__block不管是ARC还是MRC模式下都可以使用,...
    LZM轮回阅读 3,427评论 0 6
  • Object-c的类可以多重继承么?可以实现多个接口么?Category是什么?重写一个类的方式用继承好还是分类好...
    small_Sun阅读 753评论 0 0