- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
我有两个 NSManagedObjectContext
,名为 importContext
和 childContext
。 childContext
是 importContext
的 child ,它们都是 NSPrivateQueueConcurrencyType
。
为了让事情远离主线程,我在 importContext
的队列上做了很多工作。这项工作涉及大量的提取和保存,因此将整个事情包装在 importContext
的 performBlockAndWait:
中很方便(它确实需要通过同步操作,因为我在 performBlockAndWait
之后的代码取决于它的结果)。
在这项工作的某个时候,我可能需要从 JSON 结果创建新的托管对象。这些 JSON 值可能无效并且无法通过我的验证,因此在我创建对象之后,如果它们不好,我需要能够抛弃它们。这就是 childContext
的用武之地。我将我的新对象插入其中,如果它的 JSON 属性最终没有意义,我会放弃 childContext
。
当我需要保存 childContext
时,问题就来了。我希望它有自己的专用队列,与其父队列分开。但是,这只会在 iOS 7(而不是 iOS 8)上导致死锁。当我在 iOS 8 模拟器和设备上运行相同的代码时,childContext
会在单独的线程上创建自己的队列并正确保存。
当我运行 iOS 7 时,childContext
试图在父级队列中执行 save:
,但父级正在等待其子级,这会导致僵局。在 iOS 8 中,这不会发生。有谁知道为什么吗?
这是简化的代码:
-(NSManagedObjectContext *)importContext
{
NSManagedObjectContext* moc = [[NSManagedObjectContext alloc] initWithConcurrencyType:NSPrivateQueueConcurrencyType];
moc.persistentStoreCoordinator = [self storeCoordinator];
return moc;
}
-(void)updateItems:(NSArray*)ItemDescriptions
{
[self.importContext performBlockAndWait:^{
//get info and update
...
...
if(needToCreateNewItem){
NSManagedObjectContext* childContext = [[NSManagedObjectContext alloc] initWithConcurrencyType:NSPrivateQueueConcurrencyType];
childContext.parentContext = self.importedContext;
//Insert and create new item
...
[childContext performBlockAndWait:^{
id newObject = [NSEntityDescription insertNewObjectForEntityForName:[self entityName]
inManagedObjectContext:childContext];
}];
...
// Do something with this object
if([newObject isReadyToSave])
__block NSError* e = nil;
__block BOOL saveSucceeded = NO;
[childContext performBlockAndWait:^{
saveSucceeded = [childContext save:&e]; // DEADLOCK ON iOS 7!!!!
}];
}
....
}
}];
}
一个简单的解决方法是将工作保留在单独的调度队列中(而不是 importContext
的队列),但我问这个问题的原因是因为我想了解发生这种情况的根本原因。我认为 child 的保存应该只发生在自己的队列中。
更新 1
回复。马库斯的问题:
updateItems:
从操作队列中的 NSInvocationOperation
调用,因此它不在主队列中。
在 iOS 7 上,我可以随时暂停应用程序并查看堆栈,托管对象上下文的队列将死锁:
(lldb) bt
* thread #7: tid = 0xed07, 0x38546aa8 libsystem_kernel.dylib`semaphore_wait_trap + 8, queue = 'NSManagedObjectContext Queue'
frame #0: 0x38546aa8 libsystem_kernel.dylib`semaphore_wait_trap + 8
frame #1: 0x385bbbac libsystem_platform.dylib`_os_semaphore_wait + 12
frame #2: 0x3848461a libdispatch.dylib`_dispatch_barrier_sync_f_slow + 138
frame #3: 0x2d4f3df2 CoreData`_perform + 102
frame #4: 0x2d4fe1ac CoreData`-[NSManagedObjectContext(_NestedContextSupport) executeRequest:withContext:error:] + 240
frame #5: 0x2d492f42 CoreData`-[NSManagedObjectContext save:] + 826
* frame #6: 0x000c1c96 DBDevApp`__69+[DBManagedObject createWithAttributes:inManagedObjectContext:error:]_block_invoke77(.block_descriptor=<unavailable>) + 118 at DBManagedObject.m:117
frame #7: 0x2d4f6934 CoreData`developerSubmittedBlockToNSManagedObjectContextPerform + 88
frame #8: 0x3847e81e libdispatch.dylib`_dispatch_client_callout + 22
frame #9: 0x384847ca libdispatch.dylib`_dispatch_barrier_sync_f_invoke + 26
frame #10: 0x2d4f6a72 CoreData`-[NSManagedObjectContext performBlockAndWait:] + 106
frame #11: 0x000c1916 DBDevApp`+[DBManagedObject createWithAttributes:inManagedObjectContext:error:](self=0x005c1790, _cmd=0x0054a033, attributes=0x188e context=0x17500800, error=0x02e68ae8) + 658 at DBManagedObject.m:116
frame #12: 0x000fe138 DBDevApp`-[DBAPIController createOrUpdateItems:withIDs:IDKeys:ofClass:amongExistingItems:withFindByIDPredicate:](self=0x17775de0, _cmd=0x0054de newItemDescriptions=0x188eada0, itemIDs=0x18849580, idKey=0x0058e290, class=0x005c1790, existingItems=0x1756b560, findByID=0x18849c80) + 2472 at DBAPIController.m:972
frame #13: 0x00100ca0 DBDevApp`__39-[DBAPIController updatePatientGroups:]_block_invoke(.block_descriptor=0x02e68ce0) + 476 at DBAPIController.m:1198
frame #14: 0x2d4f6934 CoreData`developerSubmittedBlockToNSManagedObjectContextPerform
frame #15: 0x3847e81e libdispatch.dylib`_dispatch_client_callout + 22
frame #16: 0x384847ca libdispatch.dylib`_dispatch_barrier_sync_f_invoke + 26
frame #17: 0x2d4f6a72 CoreData`-[NSManagedObjectContext performBlockAndWait:] + 106
frame #18: 0x00100a96 DBDevApp`-[DBAPIController updatePatientGroups:](self=0x17775de0, _cmd=0x0054dfcd, groupsArray=0x188eada0) + 214 at DBAPIController.m:1191
frame #19: 0x2d721584 CoreFoundation`__invoking___ + 68
frame #20: 0x2d66c0da CoreFoundation`-[NSInvocation invoke] + 282
frame #21: 0x2e0f3d2c Foundation`-[NSInvocationOperation main] + 112
frame #22: 0x2e0515aa Foundation`-[__NSOperationInternal _start:] + 770
frame #23: 0x2e0f576c Foundation`__NSOQSchedule_f + 60
frame #24: 0x38484f10 libdispatch.dylib`_dispatch_queue_drain$VARIANT$mp + 488
frame #25: 0x38484c96 libdispatch.dylib`_dispatch_queue_invoke$VARIANT$mp + 42
frame #26: 0x38485a44 libdispatch.dylib`_dispatch_root_queue_drain + 76
frame #27: 0x38485d28 libdispatch.dylib`_dispatch_worker_thread2 + 56
frame #28: 0x385c0bd2 libsystem_pthread.dylib`_pthread_wqthread + 298
我上面显示的代码是一个简化版本。我创建新子上下文的部分位于名为 DBManagedObject
的类中。这是整个堆栈的屏幕截图:
更新 2 - 解释 DBManagedObject
DBManagedObject
是我所有核心数据类的基类。它基本上处理与 JSON 解析字典之间的转换。它有 3 个主要方法:+createWithAttributes:inManagedObjectContext:error:
、-updateWithAttributes:error:
和 attributes
。
+createWithAttributes:inManagedObjectContext:error:
:创建提供的托管对象上下文的子上下文,在子上下文中插入一个新对象并调用 updateWithAttributes:error:
在那个对象上。如果更新成功(即我们要在此对象上设置的所有值都有意义),它会保存子上下文,在作为参数传入的 MOC 中获取对新对象的引用,并返回该引用:
NSManagedObjectContext* childContext = [[NSManagedObjectContext alloc] initWithConcurrencyType:NSPrivateQueueConcurrencyType];
childContext.parentContext = context;
__block id newObject;
[childContext performBlockAndWait:^{
newObject = [NSEntityDescription insertNewObjectForEntityForName:[self entityName] inManagedObjectContext:childContext];
}];
if ([newObject updateWithAttributes:attributes error:error])
{
NSError* e = nil;
if ([childContext save:&e])
{
id parentContextObject = [context objectWithID:[(NSManagedObject*)newObject objectID]];
return parentContextObject;
}
else
{
if (error != NULL) {
*error = e;
}
return nil;
}
}
else
return nil;
updateWithAttributes:error:
:将 JSON 键之间的键转换为我在数据模型中用作实体属性的键。 (即“first_name”变为“firstName”)。如果需要,它还会格式化 JSON 值(日期字符串变为 NSDate
)。它还设置关系。
最佳答案
通过查看您的代码,我发现您有 2 个 [childContext performBlockAndWait:^{ 是嵌套的。删除其中一个应该可以解决您在 ios7 中的问题。该代码已在该线程中运行,您无需再次执行。
始终检查您是否有相同上下文的任何嵌套 performBlocks。这导致我的应用程序在 ios7 之前死锁并在 ios8 中工作
检查方法是当您看到死锁时,在调试器中按下暂停键,查看是什么阻止了所有线程正在运行。查看该特定代码并检查嵌套 block 。
关于ios - performBlockAndWait 在 iOS 7 上具有私有(private)队列死锁父级的子上下文,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26070111/
我正在创建一个 NSManagedObjectContext在私有(private)队列中处理我从文件和/或服务中获取的数据更新: NSManagedObjectContext *privateCon
Utility.managedObjectContext().performBlockAndWait({ }) dispatch_sync(dispatch_get_main_queue(), { }
上下文:使用 Core Data,我在私有(private)队列上有一个主上下文(我将调用 mainContext),在主队列上有另一个上下文,主队列的子队列(我将调用子上下文)。队列无关紧要,我的问
在核心数据最佳实践的 WWDC 2012 视频中,那个人说 performBlockAndWait: 不包含自动释放池。 (而 performBlock: 确实如此) 为什么不呢?这样做的具体后果是什
我正在编写一个函数来执行一些 CoreData 的东西。我希望函数仅在 所有 CoreData 操作执行完毕后返回。 CoreData 涉及在后台上下文中创建一个对象,然后在父上下文中执行更多操作:
我从已发布的应用程序中获得以下崩溃报告: synchronizeMyWords 方法从数据库中获取实体,创建具有主上下文父级的私有(private)队列上下文,最后保存结果。所有操作都在后台线程中。每
我遇到了无法解决的 Core Data 问题。我以艰难的方式了解了核心数据中的并发问题,因此我非常小心,只在 performBlock: 和 performBlockAndWait: block 中执
我有一个下载一些数据的 NSURLSession(带有委托(delegate))。我不希望启动下载 (refresh) 的方法在执行委托(delegate)方法之前返回。为此,我在 refresh 的
print("queue1: \(NSOperationQueue.currentQueue())") managedObjectContext.performBlockAndWait({ p
在将 NSManagedObjectContext 的 performBlock: 与通知中心一起使用时,我遇到了有趣的行为。 我从主 UI 线程触发异步数据下载(使用 NSURLConnection
我有一个 OSX 应用程序,我在其中使用父/子 NSManagedObjectContext 设置。子 MOC 具有 NSPrivateQueueConcurrencyType 并且是我主要使用的类型
我有一个 NSManagedObjectContext 声明如下: - (NSManagedObjectContext *) backgroundMOC { if (backgroundMOC
我注意到,对于 NSManagedObjectContext 和 NSMainQueueConcurrencyType 来说,performBlockAndWait 是可能的:并在除接收者的(主)队列
仍在开发我的 RSS 阅读器。 我的数据模型大致由以下 NSManagedObject 层次结构组成: Category > Feed > Post 我的应用程序使用以下内容: 具有 NSPrivat
我有两个 NSManagedObjectContext,名为 importContext 和 childContext。 childContext 是 importContext 的 child ,它
更新:我已经准备好了可以无问题地再现问题的示例,请使用以下URL下载测试项目: https://www.dsr-company.com/fm.php?Download=1&FileToDL=DeadL
我使用的是 Xcode 7.1,在打开我现有的应用程序时收到此警告。如果有人展示使用 -performBlockAndWait: 的方法将会很有帮助 谢谢 最佳答案 正如 Mundi 所说,您不需要锁
我是一名优秀的程序员,十分优秀!