3

我在 CoreData 和父子 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];
}

关联的 managedObjectContext 在同一个队列(NSMainQueueConcurrencyType)上。

否则,如果使用所有并发链managedObject创建,则返回 YES,并且所有更改都将传递给父上下文。+newObjectForInsertionsaveContext:

我不知道这是一个错误还是 CoreData 应该工作的方式。

苹果开发者论坛上的同样问题:

https://devforums.apple.com/thread/174677?tstart=90

4

1 回答 1

1

您应该使用 concurrencyType 初始化上下文:

context = [[NSManagedObjectContext alloc] initWithConcurrencyType:NSPrivateQueueConcurrencyType];

另外,设置合并策略

[context setMergePolicy:NSMergeByPropertyObjectTrumpMergePolicy];

NSMergeByPropertyObjectTrumpMergePolicy

此策略合并对象的持久存储版本与当前内存版本之间的冲突,优先考虑内存更改。合并由单个属性发生。对于在外部源和内存中都已更改的属性,内存中的更改胜过外部更改。

顺便说一句,我发现了类似的问题:奇怪的行为 -when-using-child-parent-nsmanagedobjectcontext查看使用通知合并的已接受答案。

于 2013-07-12T11:26:22.360 回答