非法尝试在不同上下文中的对象之间建立关系“检查表”

时间:2013-08-09 14:21:21

标签: ios core-data nsmanagedobjectcontext

我正在使用Core Data进行iOS应用的离线存储,并且我一直收到以下错误:

  

非法尝试在不同背景下的对象之间建立关系'清单'

问题是,我肯定使用相同的上下文来创建关系。下面是触发问题的代码块:

 NSMutableSet *checklists = [[NSMutableSet alloc] init];
for (NSDictionary *dict in array){
    Checklists *checklist = [self addChecklist:dict withContext:context];
    [checklists addObject:checklist];
    DLog(@"context: %@", checklist.managedObjectContext);
}

//Delete any lists that aren't from the server
NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
Users *currentUser = (Users *)[self getUserWithId:[defaults objectForKey:@"user_id"] withContext:context];
DLog(@"context: %@", context);
currentUser.checklists = checklists;

在最后一行currentUser.checklists = checklists抛出异常。来自context的日志是相同的,并且对象位于同一个线程中。有没有人遇到过这个?

addChecklist:withContext:的代码:

- (Checklists *) addChecklist:(NSDictionary *) dict withContext:(NSManagedObjectContext *)context{
    //Pull the list from the database and update
    Checklists *checklist = [self getChecklistWithId:dict[@"_id"] withContext:context];

    //If the list doesn't exist, add it to the database
    if(!checklist)
        checklist = (Checklists *)[NSEntityDescription insertNewObjectForEntityForName:@"Checklists" inManagedObjectContext:context];

    //Update or create the list attributes
    checklist.checklist_id = dict[@"_id"] ? dict[@"_id"] : [NSString stringWithFormat:@"%d", (arc4random() % 2000)];
    checklist.desc = dict[@"description"] == [NSNull null] ? nil : dict[@"description"];
    checklist.name = dict[@"name"] == [NSNull null] ? nil : dict[@"name"];
    checklist.published = dict[@"published"] == [NSNull null] ? 0 : dict[@"published"];
    checklist.created_at = dict[@"created_at"] == [NSNull null] ? nil : dict[@"created_at"];
    checklist.updated_at = dict[@"updated_at"] == [NSNull null] ? nil : dict[@"updated_at"];

    [self addChecklistSections:dict[@"checklist_sections"] withChecklist:checklist withContext:context];

    NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];

    [self saveMasterContext];

    return checklist;
}

getUserWithId: withContext:的代码

- (Users *) getUserWithId:(NSString *)user_id withContext:(NSManagedObjectContext *) context
{
    NSFetchRequest *request = [[NSFetchRequest alloc] init];
    NSEntityDescription *entity = [NSEntityDescription entityForName:@"Users" inManagedObjectContext:context];
    [request setEntity:entity];
    NSPredicate *pred = [NSPredicate predicateWithFormat:@"(user_id = %@)", user_id];
    [request setPredicate:pred];

    NSError *error = nil;
    NSMutableArray *users = [[context executeFetchRequest:request error:&error] mutableCopy];
    if (users == nil || users.count == 0) {
        return nil;
    }
    return users[0];
}

1 个答案:

答案 0 :(得分:1)

我没有看到制作mutableCopy获取用户的动机。我想这可能是你得到上下文错误的原因。从理论上讲,它应该只是复制指向对象的指针,但是谁知道幕后发生了什么。

相反,请将获取请求的fetchLimit设置为1,然后检查您是否获得了一个结果。直接返回此结果,而不是其副本。

当你无论如何都要传递上下文时,也不清楚为什么要调用上下文保存方法(saveMasterContext)。调用

更合乎逻辑
[context save:nil];

此外,根据Apple的大量示例代码,您应该考虑将上下文设置为类的@property,而不是将其作为方法参数传递。

-

顺便说一句,我真的认为你应该放弃以复数形式命名实体的习惯。清单实际上是一个清单,而不是清单。

此外,依赖字符串的id方案效率不高。考虑使用数字等价物。此外,我发现分配一个随机的清单ID相当危险。

此外,代码中有一些杂散行,例如在从addChecklist返回之前为用户默认值定义一个未使用的变量。