我坚持使用CoreData和Parent-Child MOC这个问题:在向子MOC添加对象时,保存它们并保存父MOC所有对象都将其属性重置为defaultValue。
我在这里粘贴了来自两个MOC的日志,特别是这些日志中的“stringAttribute”和“date”属性被重置。
我到处搜索这个问题,但我没有找到任何东西,我也看了很多实施的亲子MOC,但我无法弄清楚我做错了什么。
提前致谢!
以下是代码段:
我将一些NSManagedObject添加到主上下文中,然后使用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;
[...]
})
)}
记录1.2
Inserted objects:
{(
<Entity: 0xb53ce80> (entity: Entity; id: 0x9582d40 <x-coredata:///Entity/t24D0F98B-CB94-41D3-BEDD-79913454A9152> ; data: {
[...]
dateAttribute = "2013-01-05 11:00:00 +0000";
stringAttribute = nil;
[...]
})
)}
更新
我应该提到添加到上下文中的managedObject
是使用上下文 nil 初始化的。然后在调用saveContext:
之前,我检查是否存在object.managedObjectContext
,如果它为零,那么我将其设置为[self managedObjectContext]
,即使用上述方法创建的managedObject
。因此,如果使用nil上下文创建+ (id)newObjectForInsertion
{
return [[self alloc] initWithEntity:[self entityDescription] insertIntoManagedObjectContext:[DKDatabaseManager defaultManager].managedObjectContext];
}
,或者使用:
managedObject
关联的managedObjectContext位于同一队列中(NSMainQueueConcurrencyType)。
否则,如果使用+newObjectForInsertion
创建saveContext:
,则所有{{1}}并发链都返回YES,并且所有更改都将传递给父上下文。
我不知道这是一个错误还是CoreData的工作方式。
Apple Developer论坛上的相同问题:
答案 0 :(得分:1)
你应该使用concurrencyType初始化上下文:
context = [[NSManagedObjectContext alloc] initWithConcurrencyType:NSPrivateQueueConcurrencyType];
另外,设置合并政策
[context setMergePolicy:NSMergeByPropertyObjectTrumpMergePolicy];
NSMergeByPropertyObjectTrumpMergePolicy
此策略合并持久性商店的版本之间的冲突 对象和当前的内存中版本,优先考虑 内存中的变化。合并由个人财产发生。对于 已在外部源和内部更改的属性 内存中,内存中的变化胜过外部变化。
顺便说一句,我发现了类似的问题:strange-behavior-when-using-child-parent-nsmanagedobjectcontext查看使用通知合并的已接受答案。