1

我在 iOS 5 中有一个目录应用程序,它从 XML 下载数据并UITableView使用带有NSFetchedResultsController. 数据存储在 Core Data 中UIManagedDocument。因为我不想在下载和导入数据时阻塞主队列,所以我创建了一个用于下载数据的后台队列和一个新的子队列,NSManagedObjectContext用于作为父NSPrivateQueueConcurrencyType级导入数据。document.managedObjectContext当我完成导入数据时,我-save:在子上下文中进行了更改,并且更改会传播到父上下文。浏览目录时,我会在需要时导入其他数据。一切正常,直到UIManagedDocument自动保存。

我已打开核心数据 SQL 调试,-com.apple.CoreData.SQLDebug 1以查看文档何时自动保存。

在创建具有重复 ID 的文档的自动保存对象之后document.managedObjectContext(我所有的实体 id 数据库都有唯一的 id 参数)。

我究竟做错了什么?


我创建了一个简单的示例代码来重现该问题。
这是代码:http
://dl.dropbox.com/u/20987346/ViewController.m 这是完整的 Xcode 项目:http ://dl.dropbox.com/u/20987346/CoreDataTest.zip

下面是在后台进行导入的方法。

- (void)backgroundImport
{
    static int counter;

    NSManagedObjectContext *backgroundContext;
    backgroundContext = [[NSManagedObjectContext alloc] initWithConcurrencyType:NSPrivateQueueConcurrencyType];
    backgroundContext.parentContext = self.document.managedObjectContext;

    [backgroundContext performBlock:^{
        NSManagedObject *entity;

        for (int i = 0; i < 2; i++) {
            entity = [self entityWithID:[NSNumber numberWithInt:arc4random() % 20 + 1]
                 inManagedObjectContext:backgroundContext];
            [entity setValue:[NSString stringWithFormat:@"A Name %d", ++counter] forKey:@"name"];
        }

        [self dumpEntitiesInManagedObjectContext:backgroundContext];

        NSError *error;
        [backgroundContext save:&error];
        if (error) NSLog(@"%@ (%@)", [error localizedDescription], [error localizedFailureReason]);

        [backgroundContext.parentContext performBlock:^{
            [self dumpEntitiesInManagedObjectContext:backgroundContext.parentContext];
        }];
    }];
}

该方法导入两个实体。获取具有指定 ID 属性的-entityWithID:实体,如果它不存在,则使用NSEntityDescription -insertNewObjectForEntityForName:. 转储所有实体以-dumpEntitiesInManagedObjectContext:记录(一次在导入上下文中,一次在文档上下文中)。

问题是当文档被自动保存并完成一些额外的导入时,我在日志中得到以下内容:

[1140b] Entities: 10             [fb03] Entities: 11
[1140b]   2: A Name 1            [fb03]   2: A Name 1
[1140b]   3: A Name 4            [fb03]   3: A Name 4
[1140b]   4: A Name 8            [fb03]   4: A Name 8
[1140b]   5: A Name 12           [fb03]   5: A Name 12
[1140b]   6: A Name 10           [fb03]   6: A Name 10
[1140b]   8: A Name 6            [fb03]   8: A Name 6
[1140b] **12: A Name 11**        [fb03] **12: A Name 11**
[1140b]   13: A Name 9           [fb03] **12: A Name 5**
[1140b]   17: A Name 3           [fb03]   13: A Name 9
[1140b]   18: A Name 2           [fb03]   17: A Name 3
                                 [fb03]   18: A Name 2

导入上下文有 10 个实体,但主上下文有 11 个实体,ID 为 12 的实体是重复的。似乎旧对象没有在父上下文中修改,而是添加了。

4

2 回答 2

3

我仍然在处理所有这类事情(Core Data & UIManagedDocument 一起工作),但我认为这个问题可能会解决你的情况:Core Data managed object does not see related objects until restart Simulator

它涉及在“正常”流程之前使用以下命令强制临时 ID 永久存在: [context obtainPermanentIDsForObjects:[inserts allObjects] error:&error]

于 2012-04-19T13:23:30.183 回答
1

我遇到了这个问题并做了一个小分类NSEntityDescription来解决这个问题:

@implementation NSEntityDescription (PermanentID)
+ (id)insertNewPermanentObjectForEntityForName:(NSString *)entityName
                        inManagedObjectContext:(NSManagedObjectContext *)context
{
    id object = [self insertNewObjectForEntityForName:entityName inManagedObjectContext:context];
    NSError *error;
    if (![context obtainPermanentIDsForObjects:[NSArray arrayWithObject:object] error:&error]) {
        NSLog(@"Permanent ID not given");
    }
    if (error) {
        NSLog(@"%@", error);
    }
    return object;
}

在我的网站上有更多关于它的信息,但我已经在此处复制了该类别,因此您不必点击。

于 2012-09-25T20:04:59.393 回答