Ios 父MOC从子MOC获取带有空数据的更改
我在CoreData和父子MOC中遇到了这个问题:将对象添加到子MOC、保存它们并保存父MOC时,所有对象的属性都会重置为defaultValue 我在这里粘贴了来自两个MOC的日志,特别是这些日志中重置的“stringAttribute”和“date”属性 我到处寻找这个问题,但什么也没找到,我也查看了很多亲子MOC的实现,但我不知道我做错了什么 提前谢谢 以下是代码片段: 我将一些NSManagedObject添加到主上下文中,然后使用Ios 父MOC从子MOC获取带有空数据的更改,ios,objective-c,core-data,nsmanagedobject,nsmanagedobjectcontext,Ios,Objective C,Core Data,Nsmanagedobject,Nsmanagedobjectcontext,我在CoreData和父子MOC中遇到了这个问题:将对象添加到子MOC、保存它们并保存父MOC时,所有对象的属性都会重置为defaultValue 我在这里粘贴了来自两个MOC的日志,特别是这些日志中重置的“stringAttribute”和“date”属性 我到处寻找这个问题,但什么也没找到,我也查看了很多亲子MOC的实现,但我不知道我做错了什么 提前谢谢 以下是代码片段: 我将一些NSManagedObject添加到主上下文中,然后使用saveContext:方法保存 // Another
saveContext:
方法保存
// Another singleton method
- (void)anotherMethod
{
[...]
[self.managedObjectContext insertObject:managedObject];
NSError *error;
save = [self saveContext:&error];
[...]
}
// Database manager singleton method
- (BOOL)saveContext:(DKError *__autoreleasing *)error
{
__block BOOL save = NO;
__block NSError *internalError;
[self.managedObjectContext performBlockAndWait:^{
internalError = nil;
[self.managedObjectContext log]; // See log 1.1 below
save = [self.managedObjectContext save:&internalError];
if (!save) {
NSLog(@"Error saving data in main context");
} else {
[self.managedObjectContext.parentContext performBlock:^{
internalError = nil;
save = NO;
[self.managedObjectContext.parentContext log]; // See log 1.2 below
save = [self.managedObjectContext.parentContext save:&internalError];
if (!save) {
NSLog(@"Error saving data to disk!");
}
}];
}
}];
*error = [DKError errorWithNSError:internalError]; // Custom error class
return save;
}
父子上下文代码
- (NSManagedObjectContext *)writerObjectContext
{
if (_writerObjectContext != nil)
return _writerObjectContext;
NSPersistentStoreCoordinator *coordinator = [self persistentStoreCoordinator];
if (coordinator != nil) {
_writerObjectContext = [[NSManagedObjectContext alloc] initWithConcurrencyType:NSPrivateQueueConcurrencyType];
[_writerObjectContext setPersistentStoreCoordinator:coordinator];
}
return _writerObjectContext;
}
- (NSManagedObjectContext *)managedObjectContext
{
if (_managedObjectContext != nil) {
return _managedObjectContext;
}
_managedObjectContext = [[NSManagedObjectContext alloc] initWithConcurrencyType:NSMainQueueConcurrencyType];
[_managedObjectContext setParentContext:[self writerObjectContext]];
return _managedObjectContext;
}
日志1.1
Inserted objects:
{(
<Entity: 0x9595120> (entity: Entity; id: 0x9582d40 <x-coredata:///Entity/t24D0F98B-CB94-41D3-BEDD-79913454A9152> ; data: {
[...]
dateAttribute = "2013-07-12 10:36:31 +0000";
stringAttribute = ricercaEntity;
[...]
})
)}
关联的managedObjectContext位于同一队列(NSMainQueueConcurrencyType)上
否则,如果使用+newObjectForInsertion
创建managedObject
,则所有saveContext:
并发链返回YES,所有更改都传递到父上下文
我不知道这是一个bug还是CoreData的工作方式
苹果开发者论坛上同样的问题:
您应该使用concurrencyType初始化上下文:
context = [[NSManagedObjectContext alloc] initWithConcurrencyType:NSPrivateQueueConcurrencyType];
另外,设置合并策略
[context setMergePolicy:NSMergeByPropertyObjectTrumpMergePolicy];
NSMergeByProperty对象TrumpMergePolicy
此策略合并持久存储的版本之间的冲突
对象和内存中的当前版本,以
内存中的更改。合并按单个属性进行。对于
在外部源和中都已更改的属性
内存,内存中的变化胜过外部的变化
顺便说一句,我发现了类似的问题:看看使用通知进行合并的公认答案。您确定日志中的实体是相同的吗?dateAttribute未重置,但完全不同。是。它应该是相同的,因为传递的id是相同的(
entity:entity;id:0x9582d40
)。日期属性重置为.xcdatamodeld中设置的默认值。应该使用performBlockAndWait:
安全地传递更改,因此我认为objectID没有问题。我不完全确定您是否看到了,这两个上下文都是init和concurrencyType,writer和主上下文都是NSPrivateQueueConcurrencyType
。我只是尝试在编写器上下文上设置合并策略,但没有任何更改。我查看了链接,但一个解决方案与我做的相同(将背景设置为父级),另一个使用NSManagedObjectContextDidSaveNotification
,据我所知,该解决方案应与“old-before-5.0”父子模式一起使用。
context = [[NSManagedObjectContext alloc] initWithConcurrencyType:NSPrivateQueueConcurrencyType];
[context setMergePolicy:NSMergeByPropertyObjectTrumpMergePolicy];