NSFetchedResultsController 性能不佳

Posted

技术标签:

【中文标题】NSFetchedResultsController 性能不佳【英文标题】:Bad performance for NSFetchedResultsController 【发布时间】:2013-12-28 21:17:23 【问题描述】:

我正在尝试将 CoreData 中的项目加载到 UITableView 中。我最初的做法是简单地从我的BankInfo 实体中获取所有对象,将它们填充到一个数组中,然后使用该数组填充UITableViewCells

- (NSMutableArray *) bankInfos

    NSManagedObjectContext *context = [self managedObjectContext];

    NSFetchRequest *fetchRequest = [[NSFetchRequest alloc] init];
    NSEntityDescription *entity = [NSEntityDescription entityForName:@"BankInfo" inManagedObjectContext:context];
    [fetchRequest setEntity:entity];
    NSError *error;
    NSMutableArray *bankInfos = (NSMutableArray*)[context executeFetchRequest:fetchRequest error:&error];
    return bankInfos;
 

我听说NSFetchedResultsController 可以提高性能/内存管理,所以我尝试了一下(基本上按照Ray Wenderlich tutorial 推荐的方式实现):

- (NSFetchedResultsController *)fetchedResultsController 

    if (_fetchedResultsController != nil) 
        return _fetchedResultsController;
    

    NSFetchRequest *fetchRequest = [[NSFetchRequest alloc] init];
    NSEntityDescription *entity = [NSEntityDescription
        entityForName:@"FailedBankInfo" inManagedObjectContext:managedObjectContext];
    [fetchRequest setEntity:entity];

    NSSortDescriptor *sort = [[NSSortDescriptor alloc]
        initWithKey:@"details.closeDate" ascending:NO];
    [fetchRequest setSortDescriptors:[NSArray arrayWithObject:sort]];

    [fetchRequest setFetchBatchSize:20];

    NSFetchedResultsController *theFetchedResultsController =
        [[NSFetchedResultsController alloc] initWithFetchRequest:fetchRequest
            managedObjectContext:managedObjectContext sectionNameKeyPath:nil
            cacheName:@"Root"];
    self.fetchedResultsController = theFetchedResultsController;
    _fetchedResultsController.delegate = self;

    return _fetchedResultsController;


使用仪器分析代码后,我发现NSFetchedResultsController 将对象加载到UITableView 所需的时间大约是我最初方法的两倍。特别是这一行:

BankInfo *bankInfo = [_fetchedResultsController objectAtIndexPath:indexPath]; 

需要 292 毫秒,而加载整个 BankInfos 数组需要大约 150 毫秒。有谁知道这是为什么?

【问题讨论】:

您确定这不是过早优化的情况吗?滚动时每秒获得多少帧? @MichaelG.Emmons - 每秒帧数非常糟糕,只有 20 多岁(请注意这是 iPhone4),因为最初为每个单元格加载 BankInfos,所以非常不稳定。 【参考方案1】:

我遇到的问题与 CoreData 性能无关,但与我不小心将全尺寸图像保存/加载为表格视图中的缩略图有关。一旦我解决了这个问题,性能问题就消失了。

【讨论】:

如果您想进一步提高性能,将图像放在单独的实体中通常是个好主意。这允许核心数据在需要时对图像数据进行故障处理,并在必要时对其进行故障处理。即使我只存储缩略图,我通常也会这样做。 @MichaelG.Emmons - 感谢您提醒我这一点,记住 Apple 也在 Core Data 指南中推荐了这一点。【参考方案2】:

嗯,我们说的是ms,还是挺快的。

获取的结果控制器正在对每个单元格的 sqllite 进行查询。可以在 xcode 中开启 sqllite 调试:-com.apple.CoreData.SQLDebug 1 自己看看。 NSArray 完全在内存中填充、存储和获取。

数组和获取的控制器之间的选择不是通过考虑“速度”来完成的。

基本上,如果您有一个小对象数组,在屏幕上不可变,那么您可以安全地选择 NSArray 作为表数据源。

相反,如果您有很多对象或计划拥有越来越多的对象,也需要经常刷新,NSFetchedResultsController 是首选。

【讨论】:

+1 谢谢。问题实际上是我的错(见下文)。

以上是关于NSFetchedResultsController 性能不佳的主要内容,如果未能解决你的问题,请参考以下文章

在 Core Data 应用程序中调用 performFetch 后,是不是需要手动更新表视图?