- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我得到 uncaught exception 'NSObjectInaccessibleException', reason: 'CoreData could not fulfill a fault for '0x9f481f0 <x-coredata://ABF084FE-4BF3-4FC3-918A-BFF043589B8A/Structure/p21316>'
在 MagicalRecord 的这一行:
[[self MR_defaultContext] mergeChangesFromContextDidSaveNotification:notification];
我已经调试了 2 天了,但我仍然不确定发生了什么。
我已将其缩小到我的导入代码的一部分,该部分删除了(给定实体的)每个对象,这些对象不是刚刚导入的(因此删除了旧对象)。它看起来像这样:
- (void)deleteNonImportedEntities
{
NSString *entityName = [self.configuration entityName];
NSPredicate *predicate = [NSPredicate predicateWithFormat:@"NOT (myID IN %@)", self.resourceIds];
[MagicalRecord saveWithBlockAndWait:^(NSManagedObjectContext *localContext) {
[localContext MR_setWorkingName:@"deleteNonImportedEntities context"];
[NSClassFromString(entityName) MR_deleteAllMatchingPredicate:predicate inContext:localContext];
}];
}
非常直接。这个错误的问题是它只发生 10 次左右。如果我在这里输入太多日志,那么它就会发生得更少。但我一直在努力缩小范围。我改变了 saveWithBlockAndWait
方法显示有多少对象被插入 (i)、更新 (u) 或删除 (d)。
+ (void) saveWithBlockAndWait:(void(^)(NSManagedObjectContext *localContext))block;
{
NSManagedObjectContext *savingContext = [NSManagedObjectContext MR_rootSavingContext];
NSManagedObjectContext *localContext = [NSManagedObjectContext MR_contextWithParent:savingContext];
[localContext performBlockAndWait:^{
[localContext MR_setWorkingName:NSStringFromSelector(_cmd)];
if (block) {
block(localContext);
}
MRLogVerbose(@"Saving saveWithBlockAndWait. (i: %i, u: %i, d: %d)", [[localContext insertedObjects] count], [[localContext updatedObjects] count], [[localContext deletedObjects] count]);
[localContext MR_saveWithOptions:MRSaveParentContexts|MRSaveSynchronously completion:nil];
MRLogVerbose(@"Finished saveWithBlockAndWait");
}];
}
我也改了rootContextDidSave
(因为那是发生异常的地方)所以它会提供应该从上面的保存发送的通知中的信息(一旦它上升到根保存上下文)。
+ (void) rootContextDidSave:(NSNotification *)notification
{
if ([notification object] != [self MR_rootSavingContext])
{
return;
}
if ([NSThread isMainThread] == NO)
{
dispatch_async(dispatch_get_main_queue(), ^{
[self rootContextDidSave:notification];
});
return;
}
int inserted = [[[notification userInfo] objectForKey:@"inserted"] count];
int updated = [[[notification userInfo] objectForKey:@"updated"] count];
int deleted = [[[notification userInfo] objectForKey:@"deleted"] count];
MRLogVerbose(@"Merging changes from notification to the default context (NSManagedObjectContext+MagicalRecord.m) (i: %i, u: %i, d: %i)", inserted, updated, deleted);
[[self MR_defaultContext] mergeChangesFromContextDidSaveNotification:notification];
}
当代码没有崩溃时,日志看起来像这样:
... [278:21433] Saving saveWithBlockAndWait. (i: 0, u: 0, d: 9)
... [278:21433] → Saving <NSManagedObjectContext (0x191f8b30): deleteNonImportedEntities context> on a background thread
... [278:21433] → Save Parents? YES
... [278:21433] → Save Synchronously? YES
... [278:21433] → Saving <NSManagedObjectContext (0x1558d6e0): MagicalRecord Root Saving Context> on a background thread
... [278:21433] → Save Parents? YES
... [278:21433] → Save Synchronously? YES
... [278:21187] Merging changes from notification to the default context (NSManagedObjectContext+MagicalRecord.m) (i: 0, u: 13, d: 9)
... [278:21433] → Finished saving: <NSManagedObjectContext (0x1558d6e0): MagicalRecord Root Saving Context> on a background thread
... [278:21433] Finished saveWithBlockAndWait
我不确定为什么更新次数会随着上下文的改变而增加。
以下是它终止的时间:
... [284:22234] Saving saveWithBlockAndWait. (i: 0, u: 0, d: 8)
... [284:22234] → Saving <NSManagedObjectContext (0xa062210): deleteNonImportedEntities context> on a background thread
... [284:22234] → Save Parents? YES
... [284:22234] → Save Synchronously? YES
... [284:22234] → Saving <NSManagedObjectContext (0x17df7b60): MagicalRecord Root Saving Context> on a background thread
... [284:22234] → Save Parents? YES
... [284:22234] → Save Synchronously? YES
... [284:22234] → Finished saving: <NSManagedObjectContext (0x17df7b60): MagicalRecord Root Saving Context> on a background thread
... [284:22234] Finished saveWithBlockAndWait
All Exceptions - Breakpoint
... [284:22200] *** Terminating app due to uncaught exception 'NSObjectInaccessibleException', reason: 'CoreData could not fulfill a fault for '0x9f481f0 <x-coredata://ABF084FE-4BF3-4FC3-918A-BFF043589B8A/Structure/p21316>''
*** First throw call stack:
(0x2ae8a49f 0x389bec8b 0x2aba47dd 0x2aba3bd1 0x2aba3a35 0x2abb261d 0x2bb118c9 0x2bb1148b 0x2bb11249 0x2bb11001 0x2abb8ac5 0x2abb7cc1 0x2ac8b103 0x2ac187ad 0x2ac1894f 0x2abb7b5d 0x2ae42c61 0x2ad9e6d5 0x2bad0189 0x2abb7acf 0x2ac18433 0x2ac1864d 0x9fca3 0x9fcfb 0xc1f9db 0xc1f9c7 0xc233ed 0x2ae503b1 0x2ae4eab1 0x2ad9c3c1 0x2ad9c1d3 0x321310a9 0x2e3abfa1 0x7e821 0x38f3eaaf)
libc++abi.dylib: terminating with uncaught exception of type _NSCoreDataException
如何正确调试此问题?似乎在调用 delete 方法之前发生的导入工作正常并且被合并到默认上下文中。为什么插入/更新/删除的内容会随着通知的发送而改变?
我没有任何 View 正在监视此数据的更改。我检查了另一个 NSFetchedResultsController
s 我有而且我不相信他们被解雇了(我也有日志语句)。
更新:在通知向上发送时查看通知。当第一次调用该方法时(并且它不在主线程中),它只删除和更新了一些。但是当它到达主线程时,它只是一个巨大的插入列表。在第一次通知中被删除的那些在第二次通知中是错误的。现在只是想弄清楚为什么它们不同。
最佳答案
我已经诊断出这个问题,它与竞争条件有关。
首先介绍一下 MagicalRecord 的背景知识。它使用所有其他上下文用作父级的“保存”上下文。 “默认”上下文是用于 UI 的上下文,因此是监听通知的上下文,因此它可以合并更改。当您创建另一个上下文(或使用 MagicalRecord 的保存 block )时,它将保存上下文设置为 parent 。
问题是我的导入类中有两个主要方法。一种是根据 JSON 数据(已保存到文件中)导入所有对象,另一种是删除未导入的对象。这是每个的基本概念:
- (void)importEntities
{
...
[saveResources readJSONResponsesFromDisk:^(NSDictionary *response) {
[MagicalRecord saveWithBlockAndWait:^(NSManagedObjectContext *localContext) {
NSArray *resources = [response objectForKey:JSONkey];
NSArray *array = [NSClassFromString(entityName) MR_importFromArray:resources inContext:localContext];
[weakSelf.resourceIds addObjectsFromArray:[array valueForKey:@"myID"]];
}];
}];
}
- (void)deleteNonImportedEntities
{
...
NSPredicate *predicate = [NSPredicate predicateWithFormat:@"NOT (myID IN %@)", self.resourceIds];
[MagicalRecord saveWithBlockAndWait:^(NSManagedObjectContext *localContext) {
[NSClassFromString(entityName) MR_deleteAllMatchingPredicate:predicate inContext:localContext];
}];
}
现在我每次使用 saveWithBlockAndWait
的初衷都是因为内存问题。它循环的每个文件都可以包含 500 个实体。由于 MagicalRecord 处理导入的方式,它们每个都可以创建多个其他对象。因此,我想尽可能频繁地从内存中刷新数据(并保存到磁盘),这样它就不会堆积太多。
在进入问题的核心之前,需要了解数据在 CoreData 中的保存方式。由于 saveWithBlockAndWait
设置的上下文有一个 Saving 上下文的父级,当它被保存时,它的更改会自动合并到 Saving 上下文中。然后 Saving 保存上下文发送它的 NSManagedObjectContextDidSaveNotification
(子上下文也发送它但默认上下文忽略它)。但是,在默认上下文合并从保存上下文发送的更改之前,运行并保存了 deleteNonImportedEntities
。因此它在默认上下文之前被保存到 Saving 上下文(大约 15 次中有 1 次)。因此,当默认上下文确实尝试合并这些更改时,通知指向的某些对象没有错误(它们已在保存上下文中被删除)。因此它失败了。
现在我知道了,我已经删除了 saveWithBlockAndWait
调用并且只为整个类使用一个 localContext
。我想我将不得不重新使用自动释放池并定期保存。
更新:实际上,这会导致同样的问题。所以我想找出另一种解决方案。
更新 2:我最终采纳了下面 Aaron 的建议并将其转移到自己的操作中。我还更改了优先级以帮助将它们分开更多一点。
关于ios - 使用 mergeChangesFromContextDidSaveNotification 时 CoreData 无法完成错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/27030320/
IO 设备如何知道属于它的内存中的值在memory mapped IO 中发生了变化? ? 例如,假设内存地址 0 专用于保存 VGA 设备的背景颜色。当我们更改 memory[0] 中的值时,VGA
我目前正在开发一个使用Facebook sdk登录(通过FBLoginView)的iOS应用。 一切正常,除了那些拥有较旧版本的facebook的人。 当他们按下“使用Facebook登录”按钮时,他
假设我有: this - is an - example - with some - dashesNSRange将使用`rangeOfString:@“-”拾取“-”的第一个实例,但是如果我只想要最后
Card.io SDK提供以下详细信息: 卡号,有效期,月份,年份,CVV和邮政编码。 如何从此SDK获取国家名称。 - (void)userDidProvideCreditCardInfo:(Car
iOS 应用程序如何从网络服务下载图片并在安装过程中将它们安装到用户的 iOS 设备上?可能吗? 最佳答案 您无法控制应用在用户设备上的安装,因此无法在安装过程中下载其他数据。 只需在安装后首次启动应
我曾经开发过一款企业版 iOS 产品,我们公司曾将其出售给大型企业,供他们的员工使用。 该应用程序通过 AppStore 提供,企业用户获得了公司特定的配置文件(包含应用程序配置文件)以启用他们有权使
我正在尝试将 Card.io SDK 集成到我的 iOS 应用程序中。我想为 CardIO ui 做一个简单的本地化,如更改取消按钮标题或“在此保留信用卡”提示文本。 我在 github 上找到了这个
我正在使用 CardIOView 和 CardIOViewDelegate 类,没有可以设置为 YES 的 BOOL 来扫描 collectCardholderName。我可以看到它在 CardIOP
我有一个集成了通话工具包的 voip 应用程序。每次我从我的 voip 应用程序调用时,都会在 native 电话应用程序中创建一个新的最近通话记录。我在 voip 应用程序中也有自定义联系人(电话应
iOS 应用程序如何知道应用程序打开时屏幕上是否已经有键盘?应用程序运行后,它可以接收键盘显示/隐藏通知。但是,如果应用程序在分屏模式下作为辅助应用程序打开,而主应用程序已经显示键盘,则辅助应用程序不
我在模拟器中收到以下错误: ImageIO: CGImageReadSessionGetCachedImageBlockData *** CGImageReadSessionGetCachedIm
如 Apple 文档所示,可以通过 EAAccessory Framework 与经过认证的配件(由 Apple 认证)进行通信。但是我有点困惑,因为一些帖子告诉我它也可以通过 CoreBluetoo
尽管现在的调试器已经很不错了,但有时找出应用程序中正在发生的事情的最好方法仍然是古老的 NSLog。当您连接到计算机时,这样做很容易; Xcode 会帮助弹出日志查看器面板,然后就可以了。当您不在办公
在我的 iOS 应用程序中,我定义了一些兴趣点。其中一些有一个 Kontakt.io 信标的名称,它绑定(bind)到一个特定的 PoI(我的意思是通常贴在信标标签上的名称)。现在我想在附近发现信标,
我正在为警报提示创建一个 trigger.io 插件。尝试从警报提示返回数据。这是我的代码: // Prompt + (void)show_prompt:(ForgeTask*)task{
您好,我是 Apple iOS 的新手。我阅读并搜索了很多关于推送通知的文章,但我没有发现任何关于 APNS 从 io4 到 ios 6 的新更新的信息。任何人都可以向我提供 APNS 如何在 ios
UITabBar 的高度似乎在 iOS 7 和 8/9/10/11 之间发生了变化。我发布这个问题是为了让其他人轻松找到答案。 那么:在 iPhone 和 iPad 上的 iOS 8/9/10/11
我想我可以针对不同的 iOS 版本使用不同的 Storyboard。 由于 UI 的差异,我将创建下一个 Storyboard: Main_iPhone.storyboard Main_iPad.st
我正在写一些东西,我将使用设备的 iTunes 库中的一部分音轨来覆盖 2 个视频的组合,例如: AVMutableComposition* mixComposition = [[AVMutableC
我创建了一个简单的 iOS 程序,可以顺利编译并在 iPad 模拟器上运行良好。当我告诉 XCode 4 使用我连接的 iPad 设备时,无法编译相同的程序。问题似乎是当我尝试使用附加的 iPad 时
我是一名优秀的程序员,十分优秀!