我有一个coredata项目,我正在尝试以编程方式更新数字。 我正在从CoreData中检索对象,然后将其存储到数组中。
然后,我循环遍历该数组,以查看当前用户的IP是否存在于数据库中,并尝试更新该特定数组访问的次数。
问题是,它正在更新所有对象,而不仅仅是循环数组中的当前对象。
首先,我从核心数据中获取信息,如下所示:
- (void)fetchRecords {
// Define our table/entity to use
NSEntityDescription *entity = [NSEntityDescription entityForName:@"IPAddr" inManagedObjectContext:managedObjectContext];
// Setup the fetch request
NSFetchRequest *request = [[NSFetchRequest alloc] init];
[request setEntity:entity];
// Define how we will sort the records
NSSortDescriptor *sortDescriptor = [[NSSortDescriptor alloc] initWithKey:@"ipDate" ascending:NO];
NSArray *sortDescriptors = [NSArray arrayWithObject:sortDescriptor];
[request setSortDescriptors:sortDescriptors];
// Fetch the records and handle an error
NSError *error;
NSMutableArray *mutableFetchResults = [[managedObjectContext executeFetchRequest:request error:&error] mutableCopy];
if (!mutableFetchResults) {
// Handle the error.
// This is a serious error and should advise the user to restart the application
}
// Save our fetched data to an array
[self setIpArray: mutableFetchResults];
}
现在,我正在尝试查找当前用户IP是否存在于获取的结果中,如果存在,则更新访问的次数:
// see if the ip is present and update if necessary
-(void)ipPresent {
NSString * theCurrentIP = [self getGlobalIPAddress];
for (IPAddr *allips in ipArray)
{
if ([allips.ipNum isEqualToString:theCurrentIP]) {
NSLog(@"The IP %@ was found.", theCurrentIP);
// update the ip
NSError *error = nil;
NSNumber *ipToUpdate = allips.ipAccess;
NSNumber *addIpAccess = [[NSNumber alloc] initWithInt:1];
NSNumber *updateIpAddress = [NSNumber numberWithFloat:([ipToUpdate floatValue] + [addIpAccess floatValue])];
[self.ipArray setValue:updateIpAddress forKey:@"ipAccess"];
if ([self.managedObjectContext save:&error]) { // write to database
NSLog(@"The IP Was Updated from %@ to %@", ipToUpdate, updateIpAddress);
} else if (![self.managedObjectContext save:&error]) {
NSLog(@"failed with error: %@", error);
}
break;
} else {
NSLog(@"The IP %@ was NOT found.", theCurrentIP);
}
}
}
我很确定问题在于这一行:
[self.ipArray setValue:updateIpAddress forKey:@"ipAccess"];
同样,它正在更新所有实体,而不仅仅是当前循环中的实体。
答案 0 :(得分:2)
事实上。你使用的是错误的方法。 self.ipArray
是NSMutableArray
。
方法
- (void)setValue:(id)value forKey:(NSString *)key
用于键值编码(这使得它适用于Core Data对象),但是当应用于数组时,它将在数组中的每个条目上调用setValue:forKey:。
现在,您可以看到您也可以在单个数组元素setValue:forKey
上调用allips
,因为它的属性显然符合KVC - 否则您将遇到其他问题,而不是看到正在设定的价值。
请注意,您也可以只指定属性...
allips.ipAccess = updateIpAddress;
修改强>
对不起,可能应该读得慢一点......你明白你不必使用可变数组,对吧?您实际上并没有更改数组,只是更改数组中的元素。不可变集合意味着集合内容不能更改,但是当您有一个指向对象的指针时,只要该对象不是不可变的,您仍然可以改变其属性。
因此,如果你有一个不可变的Foo对象数组,你可以这样做......
for (Foo *foo in myImmutableArray) {
Bar *bar = [self getSomeNewBar];
[foo setBar:bar];
// If Foo is KVC compliant, you can do this too...
[foo setValue:bar for Key:@"bar"];
}
但是,如果在数组上调用setValue:forKey
,则会为数组的每个元素调用它。注意,setValue:forKey
实际上是在不可变的NSArray中声明的。
修改强>
这条评论难以理解。
核心数据对象只是另一个对象。看起来你已经将它子类化了,并为它提供了属性的属性。只需替换
[self.ipArray setValue:updateIpAddress forKey:@"ipAccess"];
与
[allips setValue:updateIpAddress forKey:@"ipAccess"];
或
allips.ipAccess = updateIpAddress;
其中任何一个都应该修改你的核心数据对象,就像任何具有名为“ipAccess”的读/写属性的对象一样
当然,假设我没有再次读错了...而且allips是你的核心数据对象......