简书上的文章已经不再维护,有兴趣阅读其他文章,或一起交流的朋友,请移步 我的博客: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