11

我或多或少有一个基本UITableViewControllerNSFetchedResultsController. UITableViewController被推入navigationController's堆栈。但是推送动画并不流畅,因为 fetchNSFetchedResultsController是在主线程上执行的,因此会阻塞 UI。

我的问题是:如何NSFetchedResultsController在后台线程中执行获取以保持动画流畅?

NSFetchedResultsController委托方法如下所示:

- (NSFetchedResultsController *)fetchedResultsController
{
    if (_fetchedResultsController != nil) {
        return _fetchedResultsController;
    }

    NSFetchRequest *fetchRequest = [[NSFetchRequest alloc] init];
    // Edit the entity name as appropriate.
    NSEntityDescription *entity = [NSEntityDescription entityForName:@"GPGrade" inManagedObjectContext:self.managedObjectContext];
    [fetchRequest setEntity:entity];

    // Set the batch size to a suitable number.
    [fetchRequest setFetchBatchSize:20];

    //Set predicate
    NSPredicate *predicate = [NSPredicate predicateWithFormat:@"parent == %@", self.subject];
    [fetchRequest setPredicate:predicate];


    // Edit the sort key as appropriate.
    NSSortDescriptor *sortDescriptor = [[NSSortDescriptor alloc] initWithKey:@"name" ascending:YES];
    NSArray *sortDescriptors = @[sortDescriptor];

    [fetchRequest setSortDescriptors:sortDescriptors];

    // Edit the section name key path and cache name if appropriate.
    // nil for section name key path means "no sections".
    NSFetchedResultsController *aFetchedResultsController = [[NSFetchedResultsController alloc] initWithFetchRequest:fetchRequest managedObjectContext:self.managedObjectContext sectionNameKeyPath:nil cacheName:@"SubjectMaster"];
    aFetchedResultsController.delegate = self;
    self.fetchedResultsController = aFetchedResultsController;

    NSError *error = nil;
    if (![self.fetchedResultsController performFetch:&error]) {
        // Replace this implementation with code to handle the error appropriately.
        // abort() causes the application to generate a crash log and terminate. You should not use this function in a shipping application, although it may be useful during development.
        NSLog(@"Unresolved error %@, %@", error, [error userInfo]);
        abort();
    }

    return _fetchedResultsController;
}

- (void)controllerWillChangeContent:(NSFetchedResultsController *)controller
{
    [self.tableView beginUpdates];
}

- (void)controller:(NSFetchedResultsController *)controller didChangeSection:(id <NSFetchedResultsSectionInfo>)sectionInfo
           atIndex:(NSUInteger)sectionIndex forChangeType:(NSFetchedResultsChangeType)type
{    
    switch(type) {
        case NSFetchedResultsChangeInsert:
            [self.tableView insertSections:[NSIndexSet indexSetWithIndex:sectionIndex] withRowAnimation:UITableViewRowAnimationFade];
            break;

        case NSFetchedResultsChangeDelete:
            [self.tableView deleteSections:[NSIndexSet indexSetWithIndex:sectionIndex] withRowAnimation:UITableViewRowAnimationFade];
            break;
    }

}

- (void)controller:(NSFetchedResultsController *)controller didChangeObject:(id)anObject
       atIndexPath:(NSIndexPath *)indexPath forChangeType:(NSFetchedResultsChangeType)type
      newIndexPath:(NSIndexPath *)newIndexPath
{    
    UITableView *tableView = self.tableView;

    switch(type) {
        case NSFetchedResultsChangeInsert:
            [tableView insertRowsAtIndexPaths:@[newIndexPath] withRowAnimation:UITableViewRowAnimationTop];
            break;

        case NSFetchedResultsChangeDelete:
            [tableView deleteRowsAtIndexPaths:@[indexPath] withRowAnimation:UITableViewRowAnimationRight];
            break;

        case NSFetchedResultsChangeUpdate:
            //[self configureCell:(GPSubjectOverviewListCell *)[tableView cellForRowAtIndexPath:indexPath] atIndexPath:indexPath];
            break;

        case NSFetchedResultsChangeMove:
            [tableView deleteRowsAtIndexPaths:@[indexPath] withRowAnimation:UITableViewRowAnimationFade];
            [tableView insertRowsAtIndexPaths:@[newIndexPath] withRowAnimation:UITableViewRowAnimationFade];
            break;
    }
}

- (void)controllerDidChangeContent:(NSFetchedResultsController *)controller
{
    [self.tableView endUpdates];
}
4

4 回答 4

19

TL;博士; 没有充分的理由在主队列上使用上下文。

可以使用 NSFetchedResultsController 在后台获取数据

绝对地。NSFetchedResultsController可以与私有队列上下文一起使用。事实上,这样做是非常快乐和高效的。当它使用私有队列时,有一个错误会阻止NSFetchedResultsController使用它的缓存,但是缓存并没有像在 iOS 3.0 中那样赢得你的青睐。设置cacheName为零,你会没事的。

1.使用NSPrivateQueueConcurrencyType. 最好不是您用于 IO 的那个。

2.使用该上下文和缓存名称 nil 创建获取的结果控制器。

3.performBlock:块中执行初始提取:

 [[[self fetchedResultsController] managedObjectContext] performBlock:^{
    NSError *fetchError = nil;
    if (![self fetchedResultsController] performFetch:&error]){
        /// handle the error. Don't just log it.
    } else {
        // Update the view from the main queue.
        [[NSOperationQueue mainQueue] addOperationWithBlock:^{
            [tableView reloadData];
         }];
    }
 }];

4.您的所有委托回调现在都将从上下文的队列中发生。如果您使用它们来更新视图,请通过像上面看到的那样分派到主队列来执行此操作。

5....

6. 利润!

您可以在此处阅读有关此内容的更多信息。

于 2014-07-29T06:50:07.460 回答
1

Core Data 的一般规则是每个线程一个托管对象上下文,每个 MOC 一个线程。考虑到这一点,您需要在主线程上执行 Fetched Results Controller 的获取,因为这是将与 FRC 的托管对象交互的线程。(请参阅核心数据编程指南 - 与核心数据并发

如果您在动画中遇到性能问题,您可能应该考虑确保在推送视图之前或之后执行获取的方法。通常你会在视图控制器中执行获取viewDidLoad:,并且导航控制器在获取完成之前不会推送视图。

于 2013-02-11T15:05:35.400 回答
0

你可以阅读这篇关于 Core Data with Multi-Threaded behavior 的非常好的文章。

希望能帮助到你..!!

于 2013-11-15T14:35:38.410 回答
0

今天解决了这个问题。一个行之有效的解决方案是将 NSFetchedResultsController(FRC) 包装在一个操作 (FRCOperation) 中。本次操作完成后的目标是保留FRC并代理他。

// After completing the operation, it will proxy calls
// from "NSFetchedResultsControllerDelegate.controller(_:didChangeContentWith:)"
protocol FetchOperationDelegate {
    func didChangeContent(snapshot: NSDiffableDataSourceSnapshotReference)
}

// Exemple Operation
protocol FetchOperation: Operation, NSFetchedResultsControllerDelegate {
    weak var delegate: FetchOperationDelegate { get set }
    
    // To get objects from NSFetchedResultsController. Calls always on the main thread
    func object(at indexPath: IndexPath) -> NSManagedObject?
}

逻辑:

  1. 打开控制器
  2. 首先创建 FetchOperation。不要忘记设置 FetchOperation.delegate = 控制器。因为执行后你应该在 FetchOperationDelegate 中得到第一个数据。
  3. 将此 FetchOperation 保存在控制器的属性中(viewModel)
  4. 执行抓取操作。NSFetchedResultsController.performFetch 在后台。不要忘记使用背景上下文。
  5. 例如 5 秒后 FetchOperation 完成
  6. 通过从操作中获取数据来更新您的控制器
  7. 从这一刻开始。您应该通过 FetchOperationDelegate(NSFetchedResultsControllerDelegate) 从 FRC 接收更新。您还可以在主线程中从 FRC 访问对象。

如果您需要更改谓词或排序。您只需要创建一个新的 FetchOperation。当新操作正在进行时,您仍将继续使用上一个 FetchOperation 版本。

于 2022-01-11T19:32:02.797 回答