NSFetchedResultsController - TableView中的不完整UI更新

时间:2012-10-14 11:55:22

标签: ios uitableview uikit nsfetchedresultscontroller

我正在使用NSFetchedResultsController刷新表格视图的数据。数据本身是通过在后台运行的XML解析器提供的。解析器完成后,它会将数据保存到自己的上下文中。 NSFetchedResultsController立即获取这些更改,并开始为每个更新的元素调用-(void)controller:didChangeObject:atIndexPath:forChangeType:newIndexPath:委托方法。这也很快,在日志文件中看起来完全正常。

但是,在-(void)controllerDidChangeContent:中,我致电UITableView的{​​{1}}。然后我在屏幕上看到更新动画,但在所有单元格中,在最后一个只有一半可见的单元格旁边,唯一可见的是单元格左侧的图像。所有文字标签都不可见。它需要大约5到10秒,然后所有标签都可见。

但是,如果我忽略了-(void)endUpdates的所有委托调用,只需在NSFetchedResultsController上调用[self.tableView reloadData],一切都可以正常运行。内容立即就在那里。

有人知道我在这里做错了什么吗?分析器显示主线程基本上什么都不做。除了调度到表视图的事件之外,还可以正确处理触摸事件。这些都没有处理。看起来表格视图正忙于做一些认真的工作,但我真的不知道那可能是什么,因为动画已经完成了。

以下是我-(void)controllerDidChangeContent:的实现:

NSFetchedResultsControllerDelegate

这是我的单元格布局的代码:

- (void)controllerWillChangeContent:(NSFetchedResultsController *)controller {
    NSLog(@"%s", __PRETTY_FUNCTION__);
    [self.tableView beginUpdates];
}

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

- (void)controller:(NSFetchedResultsController *)controller didChangeObject:(id)anObject atIndexPath:(NSIndexPath*)indexPath forChangeType:(NSFetchedResultsChangeType)type newIndexPath:(NSIndexPath*)newIndexPath {
    NSLog(@"%s", __PRETTY_FUNCTION__);
    UITableView* tableView = self.tableView;
    switch(type) {
        case NSFetchedResultsChangeInsert:
            [tableView insertRowsAtIndexPaths:[NSArray arrayWithObject:newIndexPath] withRowAnimation:UITableViewRowAnimationAutomatic];
            break;

        case NSFetchedResultsChangeDelete:
    [tableView deleteRowsAtIndexPaths:[NSArray arrayWithObject:indexPath] withRowAnimation:UITableViewRowAnimationAutomatic];
            break;

        case NSFetchedResultsChangeUpdate:
    [(NewsItemCell*)[tableView cellForRowAtIndexPath:indexPath] updateWithNews:[self.fetchedResultsController objectAtIndexPath:indexPath]];
            break;

        case NSFetchedResultsChangeMove:
            [tableView deleteRowsAtIndexPaths:[NSArray arrayWithObject:indexPath] withRowAnimation:UITableViewRowAnimationAutomatic];
            [tableView insertRowsAtIndexPaths:[NSArray arrayWithObject:newIndexPath] withRowAnimation:UITableViewRowAnimationAutomatic];
            break;
    }
}

- (void)controllerDidChangeContent:(NSFetchedResultsController *)controller {
    NSLog(@"%s", __PRETTY_FUNCTION__);
    [self.tableView endUpdates];
}

细胞的基本更新:

- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView {
    return self.fetchedResultsController.sections.count;
}

- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section {
    id<NSFetchedResultsSectionInfo> sectionInfo = [self.fetchedResultsController.sections objectAtIndex:section];
    return sectionInfo.numberOfObjects;
}

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
    News* model = [self.fetchedResultsController objectAtIndexPath:indexPath];

    NewsItemCell* cell = (NewsItemCell*)[tableView dequeueReusableCellWithIdentifier:NewsCellReuseIdentifier];
    [cell updateWithNews:model];
    cell.accessoryType = (model.content ? UITableViewCellAccessoryDisclosureIndicator : UITableViewCellAccessoryNone);
    return cell;
}

也未显示占位符字符串。标签完全是空的。只有图像可见!

3 个答案:

答案 0 :(得分:6)

让我们关注这段代码:

 case NSFetchedResultsChangeUpdate:
        [(NewsItemCell*)[tableView cellForRowAtIndexPath:indexPath] updateWithNews:[self.fetchedResultsController objectAtIndexPath:indexPath]];
        break;

带您了解正在发生的事情:

  1. 您在tableView
  2. 上打开更新事务
  3. 然后通过向tableView发出命令来响应委托调用
  4. 但是对于更新,您可以通过发出与tableView
  5. 无关的命令来区别对待
  6. 然后关闭tableView上的事务
  7. 我的意思是:tableView命令可以将更改捆绑到一个事务中。看起来您重新退回单元格的调用最终会在下一次交易中结束。所以用以下代码替换上面的代码:

    case NSFetchedResultsChangeUpdate:
            [tableView reloadRowsAtIndexPaths: [NSArray arrayWithObject:indexPath] withRowAnimation:UITableViewRowAnimationAutomatic ];
            break;
    

答案 1 :(得分:4)

我首先要检查的事情:

  1. 确保您的提取未在其他线程中执行
  2. 确保你的提取很快,如果它太慢也可能是原因

答案 2 :(得分:2)

您是否在后台线程中调用NSFetchedResultsControllerDelegate?这可能是个问题。要进行测试,您可以将bg moc的更改合并到主moc中,并观察主moc而不是bg moc。 NSManagedObjectContext的实例仅应由一个线程使用。