需要从nsarray更新nsmanagedobject for loop - iphone

时间:2012-08-18 05:53:45

标签: iphone ios nsmanagedobject

我有一个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"];

同样,它正在更新所有实体,而不仅仅是当前循环中的实体。

1 个答案:

答案 0 :(得分:2)

事实上。你使用的是错误的方法。 self.ipArrayNSMutableArray

方法

- (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是你的核心数据对象......