ios - NSMergeConflict(两个线程) - 设置合并策略无法解决

标签 ios multithreading cocoa-touch core-data conflict

好吧,这让我抓狂。

我有两个使用 UImanageddocument 的线程,一个在用户进行选择的主上下文中,具有自己的 moc 的后台线程根据时间戳将数据与服务器同步。

一切似乎都运行良好,但是当我: 1.在主上下文中添加一个对象 2.后台同步 3.从后台保存 4. 尝试再次更改同一个对象,现在从主上下文 - 主线程

我得到一个 NSMergeConflict

我将包括一些我的代码,排除大量不相关的代码,向您展示我如何初始化上下文,希望有人能启发我。我知道核心数据在这些领域很棘手。

在主线程上(在 applicationdidfinishloadingwithoptions 中):

    NSURL *url = [[[NSFileManager defaultManager] URLsForDirectory:NSLibraryDirectory inDomains:NSUserDomainMask] lastObject]; //get the default user documents folder
    url = [url URLByAppendingPathComponent:DATABASENAME];
    UIManagedDocument *doc = [[UIManagedDocument alloc] initWithFileURL:url];
    NSDictionary *options = [NSDictionary dictionaryWithObjectsAndKeys:
                             [NSNumber numberWithBool:YES], NSMigratePersistentStoresAutomaticallyOption,
                             [NSNumber numberWithBool:YES], NSInferMappingModelAutomaticallyOption, nil];
    doc.persistentStoreOptions = options;
    [doc.managedObjectContext setMergePolicy:NSMergeByPropertyObjectTrumpMergePolicy];
    self.database=doc;
    self.mainManagedObjectContext=self.database.managedObjectContext;
    [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(handleDataModelChange:) name:NSManagedObjectContextObjectsDidChangeNotification object:self.database.managedObjectContext];

之后:

- (void)handleDataModelChange:(NSNotification *)note
{
    [self save];
}

-(void) save
{
        [self.database saveToURL:self.database.fileURL forSaveOperation:UIDocumentSaveForOverwriting completionHandler:^(BOOL success) {
            batch_save=!success;
            NSLog(@"save success %d",success);
        }];
}

在后台线程上:

dispatch_queue_t fetchQ = dispatch_queue_create("syncing list", NULL);
dispatch_async(fetchQ, ^   // ***********  BACKGROUND THREAD ***********
{
    AppDelegate *delegate = (AppDelegate*)[UIApplication sharedApplication].delegate;
    NSManagedObjectContext *backgroundMOC2;
    backgroundMOC2=[[NSManagedObjectContext alloc] init];
    [backgroundMOC2 setPersistentStoreCoordinator:delegate.mainManagedObjectContext.persistentStoreCoordinator];
    [backgroundMOC2 setMergePolicy:NSMergeByPropertyObjectTrumpMergePolicy];
    [delegate.mainManagedObjectContext setMergePolicy:NSMergeByPropertyObjectTrumpMergePolicy];
    [[NSNotificationCenter defaultCenter] addObserver:delegate.mainManagedObjectContext selector:@selector(mergeChangesFromContextDidSaveNotification:) name:NSManagedObjectContextDidSaveNotification object:backgroundMOC2];


    ** pseudo-code: 
       perform fetch request from CD
       submit data to server with POST request (synchronously)
       retrieve JSON reply from server
       update what's needed in CD objects
    ** end of pseudo code

   [backgroundMOC2 save:nil];
   [[NSNotificationCenter defaultCenter] removeObserver:delegate.mainManagedObjectContext name:NSManagedObjectContextDidSaveNotification object:backgroundMOC2];
    });
dispatch_release(fetchQ);

我尝试了所有类型的合并策略常量都无济于事。

我得到这个,但文件没有保存:

conflictList = ( "NSMergeConflict (0x1a9ee1e0) for NSManagedObject (0x119aea80) with objectID '0x9dcec90 ' with oldVersion = 10 and newVersion = 11 and old object snapshot = {\n displayName = \"\";\n machineName = KIYGRDRTTDVLTQB;\n note = \"\";\n product = \"0x11967ab0 \";\n public = 1;\n published = 1;\n quantity = 3;\n registeredTo = \"\";\n registeredToEmail = \"\";\n registeredToNote = \"\";\n status = \"ON REGISTRY\";\n upDate = \"2013-03-07 10:22:01 +0000\";\n wishList = \"\";\n} and new cached row = {\n displayName = \"\";\n machineName = KIYGRDRTTDVLTQB;\n note = \"\";\n product = \"0x1a9ee3d0 \";\n public = 1;\n published = 1;\n quantity = 3;\n registeredTo = \"\";\n registeredToEmail = \"\";\n registeredToNote = \"\";\n status = \"ON REGISTRY\";\n upDate = \"2013-03-07 10:22:03 +0000\";\n wishList = \"\";\n}" ); }

顺便说一句,我在旧对象和新对象之间看到的唯一区别是指向“产品”的指针。 这可能是我的问题吗?

另一个可能的线索是,这种情况仅在添加新对象时发生,并且仅在后台同步发生之后发生。 如果我停止应用程序并重新加载它(重新加载持久性存储),我现在可以毫无问题地编辑现有对象,并根据需要多次同步它而没有问题。

谢谢大家

最佳答案

好的 friend ,

经过一个月的思索之后 - 有一个解决方案,而且没有任何我会猜到的。我实际上向 Apple 发起了支持事件,他们解决了这个问题。

所以这里是:

在为主管理对象上下文设置合并策略时,有一个属性叫做parentContext。我仍然不太清楚这个属性是什么,因为这方面的文档很少。然而,这绝对解决了我的问题。 请考虑到在我的工作流程中,我创建了一个 NSManagedDocument 并从中提取上下文。我认为大多数人都是以相反的方式来做的(大多数人不使用托管文档)。

代替

   [self.mainManagedObjectContext setMergePolicy:NSMergeByPropertyObjectTrumpMergePolicy];

我现在使用:

   NSManagedObjectContext *parentContext = delegate.mainManagedObjectContext.parentContext;
    [parentContext performBlockAndWait:^{
        [parentContext setMergePolicy:NSMergeByPropertyObjectTrumpMergePolicy];
    }];

block 事情是一种预防措施,以确保在正确的线程中执行此操作,重要的是将合并策略设置为所需上下文的 parentContext。

就个人而言,我必须补充两个大嘘声: 1. Boo to me,因为使用 NSManagedDocument 虽然它有点过时,只是因为我找到了使用它的好例子。 2. Apple 非常缺乏该领域的文档。我花了数周时间阅读有关 ManagedObjectContexts 的内容,今天是我第一次接触 parentContext 属性。

关于ios - NSMergeConflict(两个线程) - 设置合并策略无法解决,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15268881/

相关文章:

java - 为什么我不应该将 PoolingHttpClientConnectionManager 与 FutureRequestExecutionService 一起使用?

ios - Firebase 身份验证 (Swift 3) 的多个错误

ios - SDWebImage 是否同时使用内存缓存和磁盘缓存(需要时)?如果不是,我该如何处理?

Ios - 有时使用解析无法接收推送通知

c++ - 在没有 c++11 的情况下使用 boost 标记线程退出的正确方法是什么

c++ - 委托(delegate) - 来自 C++ 的 Objective-C 回调

java - 从另一个线程更新 JLabel,而无需分离线程上的所有进程?

iphone - 我可以在iTunes Connect的哪个位置上传大图像图标?

c# - 使用 Windows C# 第三方程序更改 IOS 应用程序名称

ios - UITextView 的 setText 方法仅在第二次调用该方法后才起作用。