当RKObjectManager无法发布对象时,如何丢弃一个对象中的CoreData更改

时间:2013-04-11 08:46:54

标签: ios objective-c core-data restkit

当RKObjectManager中此对象的POST / PUT请求失败时,如何丢弃对NSManagedObject实例所做的CoreData更改?

NSManagedObject *object = ...;
[object setValue:@"test" forKey: @"test"];
[[RKObjectManager sharedManager] postObject:object 
                                 parameters:nil 
                                    success:^(RKObjectRequestOperation *operation, RKMappingResult *mappingResult) { NSLog(@"success, nothing to do"); }
                                    failure:^(RKObjectRequestOperation *operation, NSError *error) { 
  // I want to discard changes made to object here 
}

更新:RestKit 0.20

3 个答案:

答案 0 :(得分:42)

我对RKObjectManager一无所知,但为了放弃对NSManagedObject实例的更改,您可以调用:

[object.managedObjectContext refreshObject:object mergeChanges:NO]

这会丢弃所有更改并将对象变为错误。

see here

答案 1 :(得分:2)

Swift 解决方案:

import CoreData

extension NSManagedObject {
    func cancelChanges() {
        managedObjectContext?.refresh(self, mergeChanges: false)
    }
}

使用:

object.cancelChanges()

答案 2 :(得分:0)

另一种解决方案是自己刷新值。

如果对一个对象有多个引用,特别是如果另一个类对属性进行KVO,则NSManagedObjectContext中的refresh方法可能会导致问题。刷新后,您将获得有故障的对象(所有属性均为零)但处于损坏状态,当您尝试访问任何属性时会收到错误消息_cd_rawData but the object is not being turned into a fault

如果您不想使用单独的上下文和NSUndoManager,则可以选择以下方法:

extension NSManagedObject {
    func revertToCommitedValues() {
        changedValues().keys.forEach { (key) in
            if let persistentValue = committedValues(forKeys: [key])[key] {
                if persistentValue is NSNull {
                    setValue(nil, forKey: key)
                }
                else {
                    setValue(persistentValue, forKey: key)
                }
            }
        }
    }
}

请注意,NSManagedObject的{​​{1}}方法不能反映关系内部的变化,因此,如果您想进行“深度重置”,则必须在每个关系上调用该方法

当然,您可以基于此方法来仅刷新某些属性。