0

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

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

4

1 回答 1

2

的确。您使用了错误的方法。 self.ipArray是一个NSMutableArray

方法

- (void)setValue:(id)value forKey:(NSString *)key

用于键值编码(这使得它适用于核心数据对象),但是当应用于数组时,它将在数组中的每个条目上调用 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是你的核心数据对象......

于 2012-08-18T06:08:23.107 回答