- 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/
SQLite、Content provider 和 Shared Preference 之间的所有已知区别。 但我想知道什么时候需要根据情况使用 SQLite 或 Content Provider 或
警告:我正在使用一个我无法完全控制的后端,所以我正在努力解决 Backbone 中的一些注意事项,这些注意事项可能在其他地方更好地解决......不幸的是,我别无选择,只能在这里处理它们! 所以,我的
我一整天都在挣扎。我的预输入搜索表达式与远程 json 数据完美配合。但是当我尝试使用相同的 json 数据作为预取数据时,建议为空。点击第一个标志后,我收到预定义消息“无法找到任何内容...”,结果
我正在制作一个模拟 NHL 选秀彩票的程序,其中屏幕右侧应该有一个 JTextField,并且在左侧绘制弹跳的选秀球。我创建了一个名为 Ball 的类,它实现了 Runnable,并在我的主 Draf
这个问题已经有答案了: How can I calculate a time span in Java and format the output? (18 个回答) 已关闭 9 年前。 这是我的代码
我有一个 ASP.NET Web API 应用程序在我的本地 IIS 实例上运行。 Web 应用程序配置有 CORS。我调用的 Web API 方法类似于: [POST("/API/{foo}/{ba
我将用户输入的时间和日期作为: DatePicker dp = (DatePicker) findViewById(R.id.datePicker); TimePicker tp = (TimePic
放宽“邻居”的标准是否足够,或者是否有其他标准行动可以采取? 最佳答案 如果所有相邻解决方案都是 Tabu,则听起来您的 Tabu 列表的大小太长或您的释放策略太严格。一个好的 Tabu 列表长度是
我正在阅读来自 cppreference 的代码示例: #include #include #include #include template void print_queue(T& q)
我快疯了,我试图理解工具提示的行为,但没有成功。 1. 第一个问题是当我尝试通过插件(按钮 1)在点击事件中使用它时 -> 如果您转到 Fiddle,您会在“内容”内看到该函数' 每次点击都会调用该属
我在功能组件中有以下代码: const [ folder, setFolder ] = useState([]); const folderData = useContext(FolderContex
我在使用预签名网址和 AFNetworking 3.0 从 S3 获取图像时遇到问题。我可以使用 NSMutableURLRequest 和 NSURLSession 获取图像,但是当我使用 AFHT
我正在使用 Oracle ojdbc 12 和 Java 8 处理 Oracle UCP 管理器的问题。当 UCP 池启动失败时,我希望关闭它创建的连接。 当池初始化期间遇到 ORA-02391:超过
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 9 年前。 Improve
引用这个plunker: https://plnkr.co/edit/GWsbdDWVvBYNMqyxzlLY?p=preview 我在 styles.css 文件和 src/app.ts 文件中指定
为什么我的条形这么细?我尝试将宽度设置为 1,它们变得非常厚。我不知道还能尝试什么。默认厚度为 0.8,这是应该的样子吗? import matplotlib.pyplot as plt import
当我编写时,查询按预期执行: SELECT id, day2.count - day1.count AS diff FROM day1 NATURAL JOIN day2; 但我真正想要的是右连接。当
我有以下时间数据: 0 08/01/16 13:07:46,335437 1 18/02/16 08:40:40,565575 2 14/01/16 22:2
一些背景知识 -我的 NodeJS 服务器在端口 3001 上运行,我的 React 应用程序在端口 3000 上运行。我在 React 应用程序 package.json 中设置了一个代理来代理对端
我面临着一个愚蠢的问题。我试图在我的 Angular 应用程序中延迟加载我的图像,我已经尝试过这个2: 但是他们都设置了 src attr 而不是 data-src,我在这里遗漏了什么吗?保留 d
我是一名优秀的程序员,十分优秀!