0

我有 90 个名为“ItemModel”的 CoreData 实体,具有 2 个属性“uid”、“description”,其中每个项目都作为 NSManagedObject 插入:

NSManagedObject *object = [NSEntityDescription insertNewObjectForEntityForName: @"ItemModel" inManagedObjectContext: AFYDelegate.managedObjectContext];

第一个服务器调用将“uid”分配给上面为键“uid”提取的 90 个项目中的每一个。上下文不保存在这里。

在稍后的第二次服务器调用中,我喜欢使用 indexPath 为每个NSManagedObject 更新 90 个项目的“描述” - 通过获取每个对象并将其传递给以下方法并保存上下文:

[self updateItemToDataModel:object withData: description];

....
....

- (void)updateItemToDataModel:(NSManagedObject *) object withData:(NSString *)data
{   
    [object setValue:data  forKey:@"description"];

    NSError * error = nil;
    if (![self.managedObjectContext save:&error]) {
       //Handle any error with the saving of the context
       NSLog(@"%@",error.localizedDescription);
    }
}

以上在关闭模拟器并再次运行代码后更新 CoreData 但效果很好,每个项目将有两个重复项,具有相同的“uid”和“描述”。这意味着我现在有 180 件商品。反复关闭和​​运行代码会创建越来越多的项目。

我尝试删除 updateItemToDataModel 方法,重置模拟器,它适用于 90 个项目。

如果有人可以提供帮助,我是 CoreData 的新手。如果我只想更新现有项目,我的代码有什么问题?

4

1 回答 1

1

您每次都将一个新对象插入到 MOC(托管对象上下文)中,而不是进行获取并查找您希望更新的对象的现有实例。

要获取现有对象,您可以像这样执行获取请求...

NSPredicate * predicate = [NSPredicate predicateWithFormat:@"uid == %@", uidToMatch];
NSFetchRequest * fetchRequest = [[NSFetchRequest alloc] init];
[fetchRequest setPredicate:predicate];
[fetchRequest setEntity:[NSEntityDescription entityForName:@"ItemModel" inManagedObjectContext:managedObjectContext]];
NSError * error = nil;
NSArray * results = [managedObjectContext executeFetchRequest:fetchRequest error:&error];
if ([results count]) {
    // you may need to handle more than one match in your code...
    // you could also set a fetch limit of 1 and guarantee you only get the first object, eg: [fetchRequest setFetchLimit:1];
}
else {
    // no results
}

您可能希望将其包装在一个辅助函数中,以便您可以重复使用它。并阅读 NSFetchRequest、NSPredicate 和编写谓词以执行更高级的获取请求。

于 2013-07-09T04:39:52.127 回答