CoreData:在返回 UITableView 之前重新加载数据

Posted

技术标签:

【中文标题】CoreData:在返回 UITableView 之前重新加载数据【英文标题】:CoreData: Reload Data Before returning to UITableView 【发布时间】:2015-03-06 01:01:51 【问题描述】:

我知道核心数据与 UITableViewController 密切相关,但是我有一个应用程序有两个页面,您可以在这些页面上更新数据,如果我能看到更新的值反映在每个页面的相关标签中,那就太好了必须返回 UITableViewController 并使用 tableView.reloadData()。有没有办法做到这一点? 谢谢

编辑:我添加了一个图表,希望能让我的问题更清楚一些。我想对视图 D 中的托管对象进行更新,我可以确认(通过 println)已经按预期工作,但是当我 poptoViewController C 时,没有任何改变,我看到的是过时的信息。只有当我返回 A(tableview)时,对对象的更改才会在 UI 中变得明显。那么有没有一种方法可以在视图 C 上重新加载对象而不必返回到 A?

编辑 #2:阅读下面的回复(谢谢!)后,我认为我的问题归结为 如何在没有 UITableView 的 ViewController 上实现 NSFetchedResultsController 的功能?

【问题讨论】:

为什么使用reloadData() 有问题?你能多解释一下你的应用是什么样的吗? 数据包含在“数据源”中,由您的 UITableViewDataSource 对象管理。它可能会或可能不会被CoreData“支持”的事实是无关紧要的。当“数据源”发生更改时,必须发出 reloadData(或其他 reload... 操作之一),以便 TableView 知道数据已更改。 【参考方案1】:

NSFetchedResultsController 几乎免费为您提供此功能。实现NSFetchedResultsControllerDelegate 并设置frc.delegate = self。当 Core Data 发生更改时,将调用适当的委托方法

下面的代码是 Xcode 为您设置的 ios Master/Detail Core Data 模板的样板。要获得所有这些的副本,请转到文件 -> 新建 -> 项目 -> 主从应用程序 -> 然后选中“使用核心数据”。

斯威夫特

func controllerWillChangeContent(controller: NSFetchedResultsController) 
    self.tableView.beginUpdates()


func controller(controller: NSFetchedResultsController, didChangeSection sectionInfo: NSFetchedResultsSectionInfo, atIndex sectionIndex: Int, forChangeType type: NSFetchedResultsChangeType) 
    switch type 
        case .Insert:
            self.tableView.insertSections(NSIndexSet(index: sectionIndex), withRowAnimation: .Fade)
        case .Delete:
            self.tableView.deleteSections(NSIndexSet(index: sectionIndex), withRowAnimation: .Fade)
        default:
            return
    


func controller(controller: NSFetchedResultsController, didChangeObject anObject: AnyObject, atIndexPath indexPath: NSIndexPath?, forChangeType type: NSFetchedResultsChangeType, newIndexPath: NSIndexPath?) 
    switch type 
        case .Insert:
            tableView.insertRowsAtIndexPaths([newIndexPath!], withRowAnimation: .Fade)
        case .Delete:
            tableView.deleteRowsAtIndexPaths([indexPath!], withRowAnimation: .Fade)
        case .Update:
            self.configureCell(tableView.cellForRowAtIndexPath(indexPath!)!, atIndexPath: indexPath!)
        case .Move:
            tableView.deleteRowsAtIndexPaths([indexPath!], withRowAnimation: .Fade)
            tableView.insertRowsAtIndexPaths([newIndexPath!], withRowAnimation: .Fade)
        default:
            return
    


func controllerDidChangeContent(controller: NSFetchedResultsController) 
    self.tableView.endUpdates()

Objective-C

- (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;

        default:
            return;
    


- (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:UITableViewRowAnimationFade];
            break;

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

        case NSFetchedResultsChangeUpdate:
            [self configureCell:[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];

【讨论】:

【参考方案2】:

回答你的问题:

如何实现 NSFetchedResultsController 的功能 没有 UITableView 的 ViewController?

您的 ViewController 应该订阅 NSManagedObjectContextDidSaveNotification 通知。查看通知的 userInfo 并处理密钥 updatedObjectsinsertedObjectsdeletedObjects。 当您的 ViewController 对任何对象感兴趣时,请更新视图。

这里is an example

【讨论】:

以上是关于CoreData:在返回 UITableView 之前重新加载数据的主要内容,如果未能解决你的问题,请参考以下文章

在 coredata 中加载图像时 UITableView 滞后

使用 coredata 在 UITableView 中添加 ABMultiValue Cell

CoreData 数据到 UIViewController 中的 UITableView?

在 CoreData 中使用 NSOrderedSet 重新排序 UITableView

在 UITableView 中显示 CoreData 对象在 Set 中排序

UITableView 与 Xcode,获取 CoreData 问题