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

你可能感兴趣的:(iOS 9中实例化NSManagedObjectContext的正确姿势 - Core Data)