3

我无法弄清楚为什么在我将数据添加到底层数据存储时,NSFetchedResultsControllerDelegate 方法没有触发。如果我重新启动 iPhone 应用程序,数据会立即显示。

我继承了 UITableViewController 并符合 NSFetchedResultsControllerDelegate:

@interface ProjectListViewController : UITableViewController <NSFetchedResultsControllerDelegate> {
    NSFetchedResultsController* fetchedResultsController_;
    NSManagedObjectContext* managedObjectContext_;
}

我实例化 NSFetchedResultsController 并将委托设置为 self:

// Controller
fetchedResultsController_ = [[NSFetchedResultsController alloc] initWithFetchRequest:request
    managedObjectContext:self.managedObjectContext 
                                                                  sectionNameKeyPath:@"Client" 
                                                                           cacheName:@"ProjectsCache"];
fetchedResultsController_.delegate = self;

我实现了委托方法:

- (void)controllerWillChangeContent:(NSFetchedResultsController*)controller {
    NSLog(@"ProjectListViewController.controllerWillChangeContent");
    // The fetch controller is about to start sending change notifications, so prepare the table view for updates.
    [self.tableView beginUpdates];
}

- (void)controllerDidChangeContent:(NSFetchedResultsController*)controller { ... }
- (void)controller:(NSFetchedResultsController*)controller didChangeObject:(id)anObject atIndexPath:(NSIndexPath *)indexPath forChangeType:(NSFetchedResultsChangeType)type newIndexPath:(NSIndexPath *)newIndexPath { ... }
- (void)controller:(NSFetchedResultsController *)controller didChangeSection:(id<NSFetchedResultsSectionInfo>)sectionInfo atIndex:(NSUInteger)sectionIndex forChangeType:(NSFetchedResultsChangeType)type { ... }

我创建了我希望保存的实体:

Project* newProject = [NSEntityDescription insertNewObjectForEntityForName:@"Project" inManagedObjectContext:self.managedObjectContext];
ProjectDetailViewController* detail = [[ProjectDetailViewController alloc] initWithStyle:UITableViewStyleGrouped 
                                                                                delegate:self 
                                                                                selector:@selector(finishedAdding:) 
                                                                                 project:newProject];

后来,我保存它:

- (void)save {
    // NSLog(@"ProjectDetailViewController.save");
    self.project.name = projectNameTextField_.text; 
    NSError* error;
    BOOL b = [self.project.managedObjectContext save:&error];
    if (!b) {
        NSLog(@"Error saving project!");
    } else {
        NSLog(@"Project was successfully saved.");
        [delegate_ performSelector:selector_ withObject:self.project];
    }
    [self dismissModalViewControllerAnimated:YES];
}

除了我的委托方法没有触发之外,这一切都很好。显然我的表格视图没有更新,查看新数据的唯一方法是显式刷新或重新启动应用程序。

我浏览了 CoreData Recipe 应用程序 - 但似乎找不到我缺少的东西。想法?

——路德

4

2 回答 2

3

如果我sectionNameKeyPath在创建原始时将 @"Client" 更改为 nil fetchedResultsController_Project实体创建保存都确实调用了委托方法!

fetchedResultsController_ = 
    [[NSFetchedResultsController alloc] initWithFetchRequest:request
                                        managedObjectContext:self.managedObjectContext 
                                          sectionNameKeyPath:nil
                                                   cacheName:@"ProjectsCache"];

经过严格审查,CoreData 食谱示例就是这样做的。随着我的数据变得越来越复杂——我想我将需要这个参数来帮助将结果分成几个部分,但是现在,很高兴看到调用了委托处理程序。

于 2009-08-24T05:05:31.323 回答
1

我对上面的阅读看起来您正在使用 2 个 NSManagedObjectContexts,一个在 ProjectListViewController 中,另一个在 ProjectDetailViewController 中(我假设它是在那里创建的,因为我没有看到它传入。

当您保存一个上下文时,它不会自动将更改传播到另一个,因此将一个保存在 ProjectDetailViewController 中不会导致更改出现在 ProjectListViewController 的上下文中,这意味着该上下文中没有任何更改告诉委托。

如果要在上下文之间推送更改,请查看 NSManagedObjectContextDidSaveNotification 和 mergeChangesFromContextDidSaveNotification:(它们是 NSManagedObjectContext文档的结尾)。

于 2009-08-24T03:20:34.060 回答