确定。这是与此处完全相同的问题:Why is NSFetchedResultsController loading all rows when setting a fetch batch size?
但他的解决办法并没有解决我的问题。
我的屏幕有几千条记录,加载速度很慢。我将批量大小设置为30(大约是屏幕上单元格的三倍),并且由于某种原因它仍然会循环并加载所有批次。
这是代码
- (NSFetchedResultsController *)guestCardFetchedResultsController
{
if (guestCardFetchedResultsController != nil) {
return guestCardFetchedResultsController;
}
// SELECT * from GuestCard
NSFetchRequest* fetchRequest = [[NSFetchRequest alloc] init];
NSEntityDescription *entity = [NSEntityDescription entityForName:@"GuestCard" inManagedObjectContext:self.context];
[fetchRequest setEntity:entity];
// ORDER BY updated DESC
NSSortDescriptor* updatedSortDescriptor = [NSSortDescriptor sortDescriptorWithKey:@"created" ascending:NO];
[fetchRequest setSortDescriptors:@[updatedSortDescriptor]];
fetchRequest.fetchBatchSize = 30;
NSString *cacheName = self.isReportProblemView ? @"reportProblemGuestCardsAll" : @"guestCardsAll";
[NSFetchedResultsController deleteCacheWithName:cacheName];
NSFetchedResultsController *aFetchedResultsController = [[NSFetchedResultsController alloc] initWithFetchRequest:fetchRequest managedObjectContext:self.context sectionNameKeyPath:@"sectionIdentifier" cacheName:cacheName];
aFetchedResultsController.delegate = self;
self.guestCardFetchedResultsController = aFetchedResultsController;
// Clean up
NSError *error = nil;
if (![[self guestCardFetchedResultsController] performFetch:&error]) {
}
return self.guestCardFetchedResultsController;
}
在这种情况下,我没有做任何非常有趣的事情。这里是一些委托代码(不包括单元格创建,我只确认了屏幕上的单元格数量):
- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView
{
if ([self.guestCardFetchedResultsController.fetchedObjects count] == 0) {
return 1;
}
// Return the number of sections.
return [[self.guestCardFetchedResultsController sections] count];
}
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
{
if ([self.guestCardFetchedResultsController.fetchedObjects count] == 0) {
return 1;
}
// Return the number of rows in the section.
id <NSFetchedResultsSectionInfo> sectionInfo = [guestCardFetchedResultsController sections][section];
return [sectionInfo numberOfObjects];
}
- (NSString *)tableView:(UITableView *)tableView titleForHeaderInSection:(NSInteger)section
{
if ([self.guestCardFetchedResultsController.fetchedObjects count] == 0) {
return @"";
}
return [[self.guestCardFetchedResultsController sections][section] name];
}
答案 0 :(得分:5)
在阅读docs on the NSFetchedResultsController initializer之后,我认为问题在于您在获取请求(已创建)中有一个排序,它不会自然地与节键路径(sectionIdentifier)排序相同。我正在查看的文档中的特定句子说:
如果此键路径与第一个排序指定的路径不同 在fetchRequest中的描述符,它们必须生成相同的相对 排序
我建议首先修改你的获取请求以对sectionIdentifier进行排序,然后创建。我认为这将解决你的问题。
NSSortDescriptor* updatedSortDescriptor = [NSSortDescriptor sortDescriptorWithKey:@"created" ascending:NO];
NSSortDescriptor* sectionSortDescriptor = [NSSortDescriptor sortDescriptorWithKey:@"sectionIdentifier" ascending:NO];
// It's critical the sectionSortDescriptor is first
[fetchRequest setSortDescriptors:@[sectionSortDescriptor, updatedSortDescriptor]];
请注意,如果created
或sectionIdentifier
属性实际上是实体类的方法,那么肯定会强制Core Data在排序/分区之前加载所有数据,因为它需要首先在每个实体上执行该方法。
答案 1 :(得分:2)
使用fetchBatchSize而不是返回NSArray中的对象时,它返回一个特殊的批处理错误数组。虽然它确实加载了所有对象ID,但是数组中的所有对象都是错误的,这意味着它们的数据未加载。然后,当您访问对象的属性时循环数组(或表重新加载),它会查询该对象的属性以及直到fetchBatchSize的下一个对象(顺便说一下,最小值似乎为4)。调试此内部行为的最佳方法是编辑方案并添加运行参数:
-com.apple.CoreData.SQLDebug 1
这将输出到控制台的各种查询。您将看到行ID的第一个查询,然后是每批数据加载。
所以你可能遇到的是加载所有行ID和创建故障对象的速度很慢,但这应该很快。
另一种可能性是,当它访问sectionIdentifier属性来构建节时,导致对象也被加载,从而导致所有批量加载。要解决此问题,您可以尝试设置propertiesToFetch以包含sectionIdentifier,这样第一个查询应该在该属性中加载,然后在访问它时不应该加载该对象。
答案 2 :(得分:-1)
我有同样的问题,最后我得出结论,它的确有效。
当你设置fetchBatchSize时,fetchRequest会将所有数据加载到缓存中,否则此参数将无用。我想它只是在加载后用于使用数据。但当你将对象作为[fetchResultController objectAtIndexPath:indexPath]
时,它由ManagedObjectContext(我想)完成,而不是由fetchRequest实现。
因此,您可以将fetchBatchSize设置为0,每次访问时都会从持久存储中请求它。或者您可以设置其他值,并在每次滚动表视图时设置fetchLimit和fetchOffset。但你仍然需要了解整体对象数。