【问题标题】:Does this method cause concurrency problems?这种方法会导致并发问题吗?
【发布时间】:2023-03-28 14:14:01
【问题描述】:

我有一个带有类方法的 CoreDataUtilities 类,该类方法可以保存托管对象上下文并处理任何错误:

+ (void)saveContext:(NSManagedObjectContext*)moc {
    NSError *error = nil;
    if (moc != nil) {
        if ([moc hasChanges] && ![moc save:&error]) {
            NSLog(@"MOC save error: %@, %@", error, [error userInfo]);
        } 
    }
}

我从 NSOperation 子类和后台线程调用这个方法,并传入线程/NSOperation 的 NSManagedObjectContext 实例。

我担心的是:

如果线程 A 调用了这个方法,当这个方法执行到一半时,线程 B 也会调用它。当然还有另一个 MOC。这会以任何方式干扰吗?从我的角度来看,它不会,因为这个方法只与调用线程“私有”或“拥有”的 MOC 实例通信。但是让我恼火的是,当多个线程同时执行同一段代码时,即使方法中的局部变量也会“混淆”。还是每个变量在新线程中都有自己的“上下文”,有自己的内存堆(或堆栈,就此而言)?

如果我把这个保存代码直接放到 NSOperation 子类和后台线程中会有很大的不同吗?为什么?

【问题讨论】:

    标签: iphone ios core-data concurrency nsmanagedobjectcontext


    【解决方案1】:

    为什么不在保存操作周围放置一个@synchronize(...) 块。这将确保托管对象上下文在已保存时不会被保存。

    + (void)saveContext:(NSManagedObjectContext*)moc {
        if (moc == nil) return;
    
        @synchronized(moc)
        {
            NSError *error = nil;
            if ([moc hasChanges] && ![moc save:&error])
                NSLog(@"MOC save error: %@, %@", error, [error userInfo]);
        } 
    }
    

    阅读here 同步。

    【讨论】:

      【解决方案2】:

      规则是每个线程必须有自己的moc。

      查看Apple's Guidelines了解更多信息。

      【讨论】:

      • 是的,每个线程都有自己的 MOC。所以如果一个线程调用 [CoreDataUtilities saveContext:myMOC] 那么会有问题吗?问题是多个线程将他们自己的 MOC 传递给这个单一的保存方法。
      • 如果您在每个线程上都有一个 moc,则无需担心重叠的保存调用,因为在 moc 之下苹果处理并发。在 moc 上进一步同步不会有任何好处,因为它会是每个线程的不同对象
      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2013-11-13
      • 1970-01-01
      相关资源
      最近更新 更多