我正在使用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;
}
也未显示占位符字符串。标签完全是空的。只有图像可见!
答案 0 :(得分:6)
让我们关注这段代码:
case NSFetchedResultsChangeUpdate:
[(NewsItemCell*)[tableView cellForRowAtIndexPath:indexPath] updateWithNews:[self.fetchedResultsController objectAtIndexPath:indexPath]];
break;
带您了解正在发生的事情:
我的意思是:tableView命令可以将更改捆绑到一个事务中。看起来您重新退回单元格的调用最终会在下一次交易中结束。所以用以下代码替换上面的代码:
case NSFetchedResultsChangeUpdate:
[tableView reloadRowsAtIndexPaths: [NSArray arrayWithObject:indexPath] withRowAnimation:UITableViewRowAnimationAutomatic ];
break;
答案 1 :(得分:4)
我首先要检查的事情:
答案 2 :(得分:2)
您是否在后台线程中调用NSFetchedResultsControllerDelegate?这可能是个问题。要进行测试,您可以将bg moc的更改合并到主moc中,并观察主moc而不是bg moc。 NSManagedObjectContext的实例仅应由一个线程使用。