NSFetchedResultsControllerDelegate vs NSFetchedResultsController

时间:2012-07-30 23:34:19

标签: ios core-data nsfetchedresultscontroller

在阅读了大量关于Core Data框架的文档和示例之后,我仍然不太了解它。会发生什么事情,我有点理解示例代码或文档的一部分,但往往不能适应更大的图片。我第二次阅读相同的代码,我仍然需要很多时间来解决它,就像第一次。这太令人沮丧了。

NSFetchedResultsControllerDelegate vs NSFetchedResultsController是核心数据中令我困惑的其中一个概念。

我认为我需要的是一个简单而概念性的解释,也许类比会有所帮助。

1 个答案:

答案 0 :(得分:5)

必须通过托管对象上下文访问所有核心数据对象。

将NSFetchRequest视为数据库查询的表示。当你告诉NSManagedObjectContext(MOC)获取数据时,你给它一个获取请求,因此它知道要获取什么。

现在,假设您有一个表格视图。您可以进行提取,并拥有所有数据,即使对于您不需要显示的内容也是如此。每次表视图更改时,您都需要重新获取数据。使用获取请求还有一些其他问题,尽管可以这样做。

要解决其中一些问题,请输入NSFetchedResultsController(FRC)。它管理数据库,以便您只拥有当时实际需要的内存中的对象。此外,它挂钩到MOC中,这样当数据库发生变化时,它会自动更改自己的数据。

因此,您创建了一个FRC并为其提供了一个获取请求。现在,它管理数据,因此它只在内存中拥有您想要的内容。但是,您需要告诉它要抓取什么,并且需要告诉您何时有数据。

这就是NSFetchedResultsControllerDelegate进来的地方。

委托是表视图(或其他组件)与FRC之间的粘合剂。委托方法是通信通道,它通知FRC要获取什么数据,何时获取数据,然后将其交给表视图。

修改

是的,FRC管理实际的数据部分。但是,当其数据发生变化时,必须通过某种方式通知正在观看数据的人。这就是代表的意思。让我们通过查看委托方法来进行不同的攻击。<​​/ p>

想象一下,数据库已经以某种方式发生了变化。 FRC通过其特殊的神奇咒语注意到了这一变化,就像一个拥有新iPhone的青少年一样,需要告诉别人。

具体而言,在您的情况下,它需要告诉负责向用户显示数据的表视图。那么,如何告诉表格视图数据已经改变了?实际上在iOS中使用了几种模式,在这种情况下,我们使用委托。

有兴趣从FRC接收此信息的人给他一个指向实现委托方法的对象的指针。当FRC想要通知感兴趣的人时,它会在作为委托给出的对象上调用适当的方法。

考虑发生了变化。 FRC代码看起来像这样(超简化但是给出了算法的想法)。

[delegate controllerWillChangeContent:self];

// Process all the changes...

for (SectionChangeInfo *info in changedSections) {
    [delegate controller:self didChangeSection:info.sectionInfo atIndex:info.index forChangeType:info.changeType];
}

for (ObjectChangeInfo *info in changedObjects) {
    [delegate controller:self didChangeObject:info.object atIndexPath:info.indexPath forChangeType:info.changeType newIndexPath:index.newIndexPath];
}

[delegate controllerDidChangeContent:self];

因此,FRC可以告诉“某人”发生变化的时间。在您的情况下,当您向FRC提供委托时,它将调用这些方法,从而使您有机会在发生更改时处理这些更改。

调用另一个委托方法来询问委托者想要用作节标题的内容。因此,假设FRC需要知道要使用什么,它会调用...

NSString *sectionTitle = [[section substringToIndex:1] uppercase];
if ([delegate respondsToSelector:@selector(controller:sectionIndexTitleForSectionName:)]) {
    sectionTitle = [delegate controller:self sectionIndexTitleForSectionName:section];
}