NSFetchedResultsController - TableView 中的 UI 更新不完整
Posted
技术标签:
【中文标题】NSFetchedResultsController - TableView 中的 UI 更新不完整【英文标题】:NSFetchedResultsController - Incomplete UI update in TableView 【发布时间】:2012-10-14 11:55:22 【问题描述】:我正在使用NSFetchedResultsController
来刷新表格视图的数据。数据本身是通过在后台运行的 XML 解析器提供的。解析器完成后,将数据保存到自己的上下文中。 NSFetchedResultsController
立即获取这些更改并开始为每个更新的元素调用 -(void)controller:didChangeObject:atIndexPath:forChangeType:newIndexPath:
委托方法。这也很快,在日志文件中看起来完全正常。
但是,在-(void)controllerDidChangeContent:
中,我调用UITableView
的-(void)endUpdates
。然后我在屏幕上看到了更新动画,但在所有单元格中,除了最后一个只有一半可见的单元格之外,唯一可见的是单元格左侧的图像。所有文本标签均不可见。大约需要 5 到 10 秒,然后所有标签都会弹出可见。
但是,如果我忽略 NSFetchedResultsController
的所有委托调用,而只是在 -(void)controllerDidChangeContent:
上调用 [self.tableView reloadData]
,那么一切正常。内容立即出现。
有人知道我在这里做错了什么吗?探查器显示主线程基本上什么都不做。除了分派到表格视图的事件之外,触摸事件也得到了正确处理。这些都不处理。看起来 table view 正忙着做一些严肃的工作,但我真的不知道那会是什么,因为动画已经完成了。
这是我对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;
以及单元格的基本更新:
- (void)updateWithNews:(News*)news
NSString* dateString = [[NSDateFormatter outputDateFormatter] stringFromDate:news.date];
self.headlineLabel.text = (news.headline ? news.headline : NSLocalizedString(@"<NewsNoHeadlineReplacement>", nil));
self.metaInfoLabel.text = [NSString stringWithFormat:NSLocalizedString(@"<NewsMetaInfoFormatDate>", nil), (dateString ? dateString : (NSLocalizedString(@"<NewsNoDateReplacement>", nil)))];
self.readIndicatorView.hidden = (news.read != nil && [news.read compare:news.parsingDate] == NSOrderedDescending);
占位符字符串也不显示。标签完全是空的。只有图像可见!
【问题讨论】:
【参考方案1】:让我们关注这段代码:
case NSFetchedResultsChangeUpdate:
[(NewsItemCell*)[tableView cellForRowAtIndexPath:indexPath] updateWithNews:[self.fetchedResultsController objectAtIndexPath:indexPath]];
break;
引导您了解正在发生的事情:
-
您在 tableView 上打开更新事务
然后您通过向 tableView 发出命令来响应委托调用
但是对于更新,您可以通过发出与 tableView 无关的命令来执行不同的操作
然后关闭 tableView 上的事务
我的意思是:tableView 命令可能会将更改捆绑到一个事务中。似乎您对重新渲染单元格的调用将在下一个事务中结束。所以将上面的代码替换为:
case NSFetchedResultsChangeUpdate:
[tableView reloadRowsAtIndexPaths: [NSArray arrayWithObject:indexPath] withRowAnimation:UITableViewRowAnimationAutomatic ];
break;
【讨论】:
这并不能解决问题。行为与以前完全相同。 嗯,您可以记录 tableView 事务和其中提交的更改吗?可能会在一个事务中加载某些数据,而在下一个事务中加载其余数据。 原来调用来自另一个线程。请参阅 Tomasz Zabłocki 的回答。【参考方案2】:我首先要检查的事情:
-
确保您的 fetch 不在其他线程中执行
确保您的抓取速度很快,如果速度太慢也可能是原因
【讨论】:
我正在主线程中进行提取,控制台显示数据已经存在。调用endUpdates
后屏幕未更新。
请在 didChangeObject 中打印您的线程 id,如下所示:NSLog(@"current = %@, main = %@", [NSThread currentThread], [NSThread mainThread]);
,验证它是同一个线程,如果不是,那是您的问题。
好的,这两个线程不一样。嗯...我知道发生了什么。感谢您一直以来的帮助。我正在合并线程上的上下文,出现通知。我认为这可能会触发同一线程中的NSFetchedResultsController
!?所以我必须保留上下文所针对的线程的一个实例,才能在这个特定的线程中调用合并?
是的,问题是在您的 bg 线程中调用了通知方法。尝试将dispatch_sync(dispatch_get_main_queue(), ^ [mainThreadContext mergeChangesFromContextDidSaveNotification:notification]; );
添加到您的通知方法中
我现在所做的是:我有一个BCAutoMergingManagedObjectContext
,它可以根据其他自动合并上下文的更改合并自身。这个上下文现在对它初始化的线程有一个弱引用,如果它还没有在其中执行,它总是同步地调度合并到这个线程中。这似乎可以解决问题。我希望这不会有任何其他副作用。感谢您的帮助!【参考方案3】:
您是否在后台线程中调用 NSFetchedResultsControllerDelegate?这可能是问题所在。为了测试,您可以将 bg moc 的更改合并到主 moc 中,并观察主 moc 而不是 bg moc。 NSManagedObjectContext 的实例应该只被一个线程使用。
【讨论】:
你说你在后台线程moc中解析。然后我假设你的 NSfrc 正在监视那个 bg moc。然后将在 bg 中调用该 frc(它们应该)。它是否正确?如果是这样,您在 bg 中使用了 frc。 不,NSFetchedResultsController
使用的是不同的 NSManagedObjectContext,但请参阅我对 Tomasz Zablocki 答案的评论。事实上,这是一个不同的线程......以上是关于NSFetchedResultsController - TableView 中的 UI 更新不完整的主要内容,如果未能解决你的问题,请参考以下文章