即使我有批量设置,NSFetchedResultsController也会加载所有行

时间:2013-09-20 00:08:19

标签: ios uitableview nsfetchedresultscontroller

确定。这是与此处完全相同的问题: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];
}

3 个答案:

答案 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]];

请注意,如果createdsectionIdentifier属性实际上是实体类的方法,那么肯定会强制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。但你仍然需要了解整体对象数。