6

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

但是,在-(void)controllerDidChangeContent:我调用UITableView-(void)endUpdates. 然后我在屏幕上看到了更新动画,但在所有单元格中,除了最后一个只有一半可见的单元格之外,唯一可见的是单元格左侧的图像。所有文本标签均不可见。大约需要 5 到 10 秒,然后所有标签都弹出可见。

但是,如果我忽略所有的委托调用NSFetchedResultsController并简单地调用[self.tableView reloadData]一切-(void)controllerDidChangeContent:正常工作。内容立即出现。

有人知道我在这里做错了什么吗?profiler 显示主线程基本上什么都不做。除了分派到表格视图的事件之外,触摸事件也得到了正确处理。这些都不处理。看起来表格视图正忙着做一些严肃的工作,但我真的不知道那会是什么,因为动画已经完成了。

这是我的实现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);
}

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

4

3 回答 3

6

让我们专注于这段代码:

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

引导您了解正在发生的事情:

  1. 你在 tableView 上打开更新事务
  2. 然后你通过向 tableView 发出命令来响应委托调用
  3. 但是对于更新,您可以通过发出与 tableView 无关的命令来执行不同的操作
  4. 然后你关闭 tableView 上的事务

我的意思是:tableView 命令可能会将更改捆绑到一个事务中。似乎您对重新渲染单元格的调用将在下一个事务中结束。所以将上面的代码替换为:

case NSFetchedResultsChangeUpdate:
        [tableView reloadRowsAtIndexPaths: [NSArray arrayWithObject:indexPath] withRowAnimation:UITableViewRowAnimationAutomatic ];
        break;
于 2012-10-23T19:19:18.557 回答
4

我首先要检查的事情:

  1. 确保您的 fetch 没有在不同的线程中执行
  2. 确保您的获取速度很快,如果速度太慢也可能是原因
于 2012-10-22T09:36:41.950 回答
2

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

于 2012-10-22T22:20:22.123 回答