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 性能不佳的主要内容,如果未能解决你的问题,请参考以下文章