NSFetchedResultsController不会自动重新加载数据

时间:2014-05-28 09:56:04

标签: ios objective-c core-data xcode5 nsfetchedresultscontroller

我已经坚持了几个小时的bug,我希望有人可以帮助我。

我有一个视图控制器,它使用NSFetchedResultsController(我正在使用核心数据)加载数据,该视图控制器获取过去具有“日期”的所有对象,即日期<现在。我希望这些数据能够在用户眼前重新加载,因此当任务移动到“过时”时,我正在使用NSNotificationCenter来触发表的重新加载。我还会在对象日期触发时发送本地通知,然后它会移动到我刚才解释的“完整”视图控制器中。

这是奇怪的部分...... 当应用程序来自后台时,我切换到“完整对象”选项卡,它们不显示。但是,如果我立即关闭应用程序并重新启动它,它们看起来都很完美。如果我在“完成对象”选项卡上进行重新加载时会发生同样的事情,但是如果我在另一个选项卡上重新加载并切换到它时,它看起来都很完美。

据我所知,我的NSFetchedResultsController不会正确跟踪对托管对象上下文所做的更改。我确定这是一个简单的错误,但我似乎无法找到它!请帮忙。

- (void)viewDidLoad {
    [super viewDidLoad];

    // Setup table view
    self.tableView = [[UITableView alloc] init];
    self.tableView.separatorInset = UIEdgeInsetsZero;
    self.tableView.tableFooterView = [[UIView alloc] init];
    self.tableView.dataSource = self;
    self.tableView.delegate = self;
    self.tableView.backgroundColor = [UIColor clearColor];
    [self.view addSubview:self.tableView];

    NSError *error = nil;
    [self.fetchedResultsController performFetch:&error];
    if (error) {
        NSLog(@"Error: %@", error.localizedDescription);
    }
}

#pragma mark - NSFetchedResultsController section

- (NSFetchRequest *)myFetchRequest {
    NSFetchRequest *fetchRequest = [NSFetchRequest fetchRequestWithEntityName:@"DBTask"];
    fetchRequest.sortDescriptors = @[[NSSortDescriptor sortDescriptorWithKey:@"title" ascending:YES]];

    NSPredicate *datePredicate = [NSPredicate predicateWithFormat:@"taskDate < %@", [NSDate date]];
    NSPredicate *completePredicate = [NSPredicate predicateWithFormat:@"state == %d", DBTaskStateComplete];
    NSPredicate *latePredicate = [NSPredicate predicateWithFormat:@"state == %d", DBTaskStateLate];
    NSPredicate *predicate = [NSCompoundPredicate orPredicateWithSubpredicates:[NSArray arrayWithObjects:datePredicate, completePredicate, latePredicate, nil]];

    [fetchRequest setPredicate:predicate];
    fetchRequest.sortDescriptors = @[[NSSortDescriptor sortDescriptorWithKey:@"taskDate" ascending:YES]];
    return fetchRequest;
}

- (NSFetchedResultsController *)fetchedResultsController {
    if (_fetchedResultsController != nil) {
        return _fetchedResultsController;
    }

    DBCoreDataStack *coreDataStack = [DBCoreDataStack defaultStack];
    NSFetchRequest *fetchRequest = [self myFetchRequest];
    _fetchedResultsController = [[NSFetchedResultsController alloc] initWithFetchRequest:fetchRequest managedObjectContext:coreDataStack.managedObjectContext sectionNameKeyPath:@"sectionName"  cacheName:nil];
    _fetchedResultsController.delegate = self;
    return _fetchedResultsController;
}

- (void)controllerWillChangeContent:(NSFetchedResultsController *)controller {
    [self.tableView beginUpdates];
}

- (void)controller:(NSFetchedResultsController *)controller didChangeObject:(id)anObject atIndexPath:(NSIndexPath *)indexPath forChangeType:(NSFetchedResultsChangeType)type newIndexPath:(NSIndexPath *)newIndexPath {
    switch (type) {
        case NSFetchedResultsChangeInsert:
            [self.tableView insertRowsAtIndexPaths:@[newIndexPath] withRowAnimation:UITableViewRowAnimationAutomatic];
            break;
        case NSFetchedResultsChangeDelete:
            [self.tableView deleteRowsAtIndexPaths:@[indexPath] withRowAnimation:UITableViewRowAnimationAutomatic];
            break;
        case NSFetchedResultsChangeUpdate:
            [self.tableView reloadRowsAtIndexPaths:@[indexPath] withRowAnimation:UITableViewRowAnimationAutomatic];
            break;
    }
}

- (void)controller:(NSFetchedResultsController *)controller didChangeSection:(id<NSFetchedResultsSectionInfo>)sectionInfo atIndex:(NSUInteger)sectionIndex forChangeType:(NSFetchedResultsChangeType)type {
    switch (type) {
        case NSFetchedResultsChangeInsert:
            [self.tableView insertSections:[NSIndexSet indexSetWithIndex:sectionIndex] withRowAnimation:UITableViewRowAnimationAutomatic];
            break;
        case NSFetchedResultsChangeDelete:
            [self.tableView deleteSections:[NSIndexSet indexSetWithIndex:sectionIndex] withRowAnimation:UITableViewRowAnimationAutomatic];
    }
}

- (void)controllerDidChangeContent:(NSFetchedResultsController *)controller {
    [self.tableView endUpdates];
}

1 个答案:

答案 0 :(得分:0)

这里的问题是日期谓词中的日期[NSDate date]是在谓词初始化时生成的

该谓词不是在当前日期再次主动跟踪taskDate,而是在创建谓词的日期。

使用获取的结果控制器委托可能不是实现此目的的最佳方法,您可以使用计时器,重置谓词并在每次计时器触发时执行新的提取。