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 并处理密钥 updatedObjects
、insertedObjects
和 deletedObjects
。
当您的 ViewController 对任何对象感兴趣时,请更新视图。
这里is an example
【讨论】:
以上是关于CoreData:在返回 UITableView 之前重新加载数据的主要内容,如果未能解决你的问题,请参考以下文章
在 coredata 中加载图像时 UITableView 滞后
使用 coredata 在 UITableView 中添加 ABMultiValue Cell
CoreData 数据到 UIViewController 中的 UITableView?
在 CoreData 中使用 NSOrderedSet 重新排序 UITableView