核心数据后台线程不更新记录

时间:2012-05-01 21:24:15

标签: ios core-data grand-central-dispatch

我在后台GCD线程中遇到Core Data的问题...我想更新一条记录,但是在获取它并设置它之后它似乎并没有真正保存更新的记录。

isUpdate是我设置的BOOL,告诉我是否正在运行第一次解析/保存,或者它是否是我需要更新的记录。在我的情况下,当我更新记录时,它实际上似乎没有在我的商店中更新。

我正在使用MagicalRecord助手。这是我的代码:

// Create background context
NSManagedObjectContext *backgroundContext = [[NSManagedObjectContext alloc] init];
[backgroundContext setPersistentStoreCoordinator:[NSPersistentStoreCoordinator defaultStoreCoordinator]];
// Save the background context and handle the save notification
[[NSNotificationCenter defaultCenter] addObserver:self
                                         selector:@selector(backgroundContextDidSave:)
                                             name:NSManagedObjectContextDidSaveNotification
                                           object:backgroundContext];

// Parsing data...
//..

Record *record;
if (!isUpdate) {
    record = [NSEntityDescription insertNewObjectForEntityForName:@"Record" inManagedObjectContext:backgroundContext];
} else {
    NSPredicate *recordPredicate = [NSPredicate predicateWithFormat:@"SELF.tag == %@", [[node attributeForName:@"tag"] stringValue]];
    record = [Record findFirstWithPredicate:recordPredicate];
}
[record setTitle:[[recordNode attributeForName:@"title"] stringValue]];

// Parsing other data...
//..

NSError *error = nil;
// save the context
[backgroundContext save:&error];
if (error) {
    NSLog(@"An error occurred: %@", error);
}

这是通知:

- (void)backgroundContextDidSave:(NSNotification *)notification {
    // Make sure we're on the main thread when updating the main context
    if (![NSThread isMainThread]) {
        [self performSelectorOnMainThread:@selector(backgroundContextDidSave:)
                           withObject:notification
                        waitUntilDone:NO];
        return;
    }
    // merge in the changes to the main context on the main thread
    [[NSManagedObjectContext defaultContext] mergeChangesFromContextDidSaveNotification:notification];
}

2 个答案:

答案 0 :(得分:2)

你的代码听起来很奇怪。

为什么在后台线程中注册NSManagedObjectContextDidSaveNotification通知?也许我错了,但您需要在应用程序的不同位置注册该通知。

如果您想让它工作,您可以在主线程中注册该通知。例如,您可以在AppDelegate

中执行此操作

例如,你可以didFinishLaunchingWithOptions:方法

[[NSNotificationCenter defaultCenter] addObserver:self
                                         selector:@selector(backgroundContextDidSave:)
                                             name:NSManagedObjectContextDidSaveNotification
                                           object:backgroundContext];

然后,始终在AppDelegate内,您可以将更改与您编写的方法合并:

- (void)backgroundContextDidSave:(NSNotification *)notification {
    // Make sure we're on the main thread when updating the main context
    if (![NSThread isMainThread]) {
        [self performSelectorOnMainThread:@selector(backgroundContextDidSave:)
                           withObject:notification
                        waitUntilDone:YES];
        return;
    }
    // merge in the changes to the main context on the main thread
    [[NSManagedObjectContext defaultContext] mergeChangesFromContextDidSaveNotification:notification];
}

代码执行以下步骤:

  1. 首先检查您是否在主线程中运行。
  2. 由于您注册的通知可能来自某个帖子 与主要的不同,你需要在上面执行选择器 主线。
  3. 最后与包含该通知的通知执行合并 你在后台做的改变。
  4. 完成后,您可以看到主要上下文已使用另一个中所做的更改进行更新。

    修改

    也许您也可以尝试将waitUntilDone更改为YES。

    希望它有所帮助。

答案 1 :(得分:1)

你混合了两种情境。这段代码可能不好:

 record = [Record findFirstWithPredicate:recordPredicate];

我认为这会在不同的上下文中找到记录而不是您的backgroundContext。你应该把它改成这样的东西:

 record = [Record findFirstWithPredicate:recordPredicate inManagedObjectContext:backgroundContext];