ios - 核心数据 : Error when deleting/adding objects

标签 ios core-data

有时,当我想通过下载和解析 json 文件来更新我的核心数据文件时,我的应用程序会崩溃。我收到以下错误:

CoreData: error: Serious application error. Exception was caught during Core Data change processing. This is usually a bug within an observer of NSManagedObjectContextObjectsDidChangeNotification. -[__NSCFSet addObject:]: attempt to insert nil with userInfo (null)

如果我在迭代期间更改属性,那么在迭代中保存 NSManagedObjectContext 的位置是否重要?

这是我的代码:

- (void) updateData
{           
    dispatch_queue_t serialdQueue;
    serialdQueue = dispatch_queue_create("update", NULL);
    dispatch_async(serialdQueue, ^{
        [self method1];
    });
    dispatch_async(serialdQueue, ^{
        [self method2];
    });
    dispatch_async(serialdQueue, ^{
        [self method3];
    });
    dispatch_async(serialdQueue, ^{
        [self method4];
    });
    dispatch_async(serialdQueue, ^{
        [self method5];
    });

}

-(void)method1
{        
    //DOWNLOAD JSON FILE
}

-(void)method2 //here i add objects to the core data file
{        
    @try {

        for (NSDictionary *jsonActivity in [json objectForKey:@"Activities"]) {  //ITERATE THROUGH JSON ACTIVITY ARRAY

            NSFetchRequest *request = [[NSFetchRequest alloc] init];

            NSEntityDescription *entity = [NSEntityDescription entityForName:@"Activity" inManagedObjectContext:self.managedObjectContext];
            [request setEntity:entity];

            NSPredicate *searchFilter = [NSPredicate predicateWithFormat:@"title == %@", [jsonActivity objectForKey:@"title"]]; // CHECK IF OBJECT FROM JSON FILE ALREADY EXISTS...
            [request setPredicate:searchFilter];
            NSError *error = nil;

            NSArray *results = [self.managedObjectContext executeFetchRequest:request error:&error];
            if (error) {
                NSLog(@"Error %@", error);
                abort();
            }

            if ([results count] == 0) { // ADD NEW ACTIVITY IF OLD LIST DOESNT CONTAIN IT

                Activity *activity = [NSEntityDescription insertNewObjectForEntityForName:@"Activity" inManagedObjectContext:self.managedObjectContext];

                activity.title = [jsonActivity objectForKey:@"title"];

                activity.remove = [NSNumber numberWithBool:NO]; // REMOVE FLAG = NO BECAUSE NEW OBJECTS AREN'T REMOVED

            } else  {

                Activity *activity = (Activity*) [results objectAtIndex:0];
                activity.remove = [NSNumber numberWithBool:NO]; // IF OBJECT ALREADY EXISTS IT SHOULD BE OBTAINED
            }

            AppDelegate *appDelegate = [[UIApplication sharedApplication] delegate];
            [appDelegate saveContext]; // SAVE MO CONTEXT
        }

    }   @catch (NSException *exception) {

        NSLog(@"Exception: %@", exception);
    }
}

-(void)method3 // DELETE OLD OBJECTS 
{   
    NSFetchRequest *fetchRequest = [[NSFetchRequest alloc] init];
    NSEntityDescription *entityDescription = [NSEntityDescription entityForName:@"Activity" inManagedObjectContext:self.managedObjectContext];
    [fetchRequest setEntity:entityDescription];

    NSPredicate *searchFilter = [NSPredicate predicateWithFormat:@"remove == %@", [NSNumber numberWithBool:YES]];
    [fetchRequest setPredicate:searchFilter];

    NSArray *objectsToDelete = [[NSArray alloc] init];

    NSError *error = nil;
    objectsToDelete = [self.managedObjectContext executeFetchRequest:fetchRequest error:&error];
    if (error) {
        NSLog(@"Error %@", error);
        abort();
    }

    for (Activity *activity in objectsToDelete) { // DELETE OBJECTS WITH THE PROPERTY REMOVE = YES

        [self.managedObjectContext deleteObject:activity]; // DELETE ACTIVITY
    }

    AppDelegate *appDelegate = [[UIApplication sharedApplication] delegate];
    [appDelegate saveContext]; // SAVE MO CONTEXT
}

-(void)method4 // CHANGE THE REMOVE PROPERTY TO YES OF ALL OBJECTS
{        
    NSFetchRequest *fetchRequest = [[NSFetchRequest alloc] init];
    NSEntityDescription *entityDescription = [NSEntityDescription entityForName:@"Activity" inManagedObjectContext:self.managedObjectContext];
    [fetchRequest setEntity:entityDescription];

    NSArray *objects = [[NSArray alloc] init];

    NSError *error = nil;
    objects = [self.managedObjectContext executeFetchRequest:fetchRequest error:&error];
    if (error) {
        NSLog(@"Error %@", error);
        abort();
    }

    for (Activity *activity in objects) {

        activity.remove = [NSNumber numberWithBool:YES];
    }

    AppDelegate *appDelegate = [[UIApplication sharedApplication] delegate];
    [appDelegate saveContext]; // SAVE MO CONTEXT

    NSLog(@"End Update");
}

-(void)method5 //UPDATE UI
{
    //UI UPDATES
}

最佳答案

您不应该访问 serialQueue 上的托管对象上下文。查看 NSManagedObjectContext documentation 的并发部分.

如果在您的代码中您的上下文正在使用 NSPrivateQueueConcurrencyTypeNSMainQueueConcurrencyType 并发类型,您可以使用一种基于 block 的方法来确保您在正确的队列中:

//asyncrhonous
[self.managedObjectContext performBlock:^{
    //do stuff with the context
}];

//syncrhonous
[self.managedObjectContext performBlockAndWait:^{
    //do stuff with the context
}];

关于ios - 核心数据 : Error when deleting/adding objects,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19184262/

相关文章:

ios - 无法运行使用iOS SDK 4.3编译的iOS 3.X应用

iPhone核心数据: Cascading delete across a many-to-one relationship

uitableview - 为什么我在尝试快速加载 tableview 时会收到 EXC_BAD_INSTRUCTION 错误?

ios - 按姓氏分组的 Core Data 家族成员

swift - NSBatchUpdateRequest : . 更新未在 didChangeObject 中触发

swift - 如何将任何值与 Coredata 实体的 Z_PK 进行比较?

iOS LayoutIfNeeded() 未返回正确的大小

ios - 在未连接 Xcode 的情况下在设备上测试应用程序状态保存/恢复

ios - didReceiveLocalNotification下慢慢打开 map app

ios - 快速将多个图像添加到单个 UIButton