Core Data应用开发实践指南
上QQ阅读APP看书,第一时间看更新

2.13 获取托管对象

想操作托管对象上下文中的现有数据,就必须先把它获取(fetch)过来。假如待获取的数据没有放在上下文里,那么Core Data会从底层的持久化存储区里把它拿出来,这个过程对开发者来说是透明的。要执行获取操作,就得有NSFetchRequest实例,该实例会返回NSArray,这个数组里面的元素都是托管对象。在执行获取操作的时候,NSFetchRequest会根据特定的实体,把每个托管对象都放在NSArray这个数组中,并将其返回给调用者。用SQL数据库领域的术语来说,获取操作类似于SELECT语句。相关代码如程序清单2-4所示。

程序清单2-4 AppDelegate.m文件的demo方法(演示如何用NSFetchRequest获取数据)

请按照下列步骤修改Grocery Dude,以便获取Item实体的所有实例:

把程序清单2-4中的代码添加到AppDelegate.m文件的demo方法里面。

运行程序并注意控制台中输出的日志,你将会看到12行非常相似的信息,每一行信息都代表从数据库中取出的一个托管对象。假如你的数据库里有重复数据,那么具体的行数可能和书中列出的有所不同。图2-10演示了正常运行程序时所产生的结果。

图2-10 获取到的托管对象

如果想查看每个托管对象的特性值,那么可以用for循环来遍历NSArray,并用NSLog把每个item的name特性值打印出来。相关代码如程序清单2-5所示。

程序清单2-5 AppDelegate.m文件中的demo方法(在获取到的托管对象中查看name特性)

请按下列步骤修改Grocery Dude,以便在控制台的日志中显示出每个item的name:

1.把AppDelegate.m文件的demo方法改为程序清单2-5的样子。方法开头的NSLog语句可以保留,也可以删掉。

重新运行应用程序,并观察控制台中的日志,现在应该可以看到每个托管对象的name特性值了。正常的运行效果如图2-11所示。

图2-11 获取到的各托管对象所具备的name特性值

2.13.1 对获取请求的结果进行排序

NSFetchRequest执行完毕之后,会返回NSArray,而NSArray本身就支持对其中的元素进行排序。另外,我们也可以换一种办法,就是给NSFetchRequest配置排序描述符(sort descriptor),这样的话,NSFetchRequest就可以直接按特定方式对获取到的托管对象进行排序了。这个排序描述符是作为NSSortDescriptor的实例传给NSFetchRequest的。用SQL数据库的术语来说,排序描述符就类似于ORDER BY语句。程序清单2-6列出了相关代码。

程序清单2-6 AppDelegate.m文件中的demo方法(演示排序功能)

请按下列步骤修改Grocery Dude,以便对获取到的托管对象进行排序:

1.修改AppDelegate.m文件的demo方法,用程序清单2-6中的粗体代码把原有的相关代码替换掉。

再次运行应用程序,并查看控制台中的日志,你会发现托管对象已经按照其名称的字母顺序排列好了。正常的运行效果如图2-12所示。

图2-12 获取到的托管对象已按名称排好顺序

2.13.2 对获取请求的结果进行筛选

有时我们并不想把与某个实体有关的全部对象都获取过来,这时可以通过谓词来筛选。我们采用NSPredicate实例来定义谓词,并将其传给NSFetchRequest实例。有了谓词之后,获取请求就会根据谓词中的标准来限定获取到的托管对象的数量。由于谓词与具体的持久化存储区无关,所以不论后端采用何种存储区,都可以使用相同的谓词来筛选它。但是,在特殊情况下,某些谓词无法适用于特定格式的存储区。比方说,matches操作符可以对in-memory存储区执行筛选,却不能用在SQLite格式的存储区上。以SQL数据库的术语来说,谓词类似于WHERE子句。

在获取请求的执行过程中,系统要根据每个托管对象来对谓词分别求值。谓词的求值结果是个Boolean值。如果是YES,那就表明该托管对象符合谓词中的标准,于是也就可以留在最终的获取结果中;但若是NO,则表示该对象不符合谓词中的标准,于是就会从最终的获取结果中剔除。

执行完NSFetchRequest之后,获取到的结果会存放在NSArray里面,此时就可以按照自己的需要对这个数组里面的托管对象进行筛选了。可以用NSArray的filteredArrayUsingPredicate方法来筛选,也可以用NSMutableArray的filter-UsingPredicate方法执行就地筛选。

以Grocery Dude程序为例,假设我们现在要把名称为Coffee的货品排除掉。在创建传递给NSFetchrequest的NSPredicate时,我们可以指明name不等于字符串Coffee。这个谓词写成代码就是name!=@“Coffee”。由于谓词支持变量替换,所以我们可以像程序清单2-7中的粗体代码那样,等到运行期再向谓词传入字符串。构建谓词所需的逻辑有时比较简单,有时却相当复杂,这要根据需求来定。有关谓词的更多信息,请访问http://developer.apple.com/网站并搜索Predicate Programming Guide。

程序清单2-7 AppDelegate.m文件中的demo方法(演示筛选功能)

请按下列步骤修改Grocery Dude,向其中添加谓词,以便筛选获取到的托管对象:

1.修改AppDelegate.m文件的demo方法,用程序清单2-7中的粗体代码把原有的相关代码替换掉。

运行应用程序,并观察控制台中的日志,你应该会发现:在列出的货品名称中,已经没有Coffee这一项了。正常的运行结果如图2-13所示。

图2-13 对获取到的托管对象进行排序和筛选,然后打印其名称

2.13.3 获取请求模板

假如每次获取托管对象时都要手工编写谓词格式确实很累人,幸好Xcode的Data Model Designer有预定义获取请求的功能。这些可复用的模板比谓词更容易配置,而且还能减少重复代码。只需根据应用程序的模型来操作一系列下拉列表框及文本框,即可配置好一份获取请求模板。但如果要自定义AND、OR这样的逻辑组合,那么这个模板就无法满足要求了,此时仍然需要通过代码来指定谓词。

请按下列步骤修改Grocery Dude,以创建获取请求模板:

1.选中Model.xcdatamodeld。

2.点击Editor>Add Fetch Request菜单项。

3.把获取请求模板的名称设为Test。

4.点击“+”按钮来配置名为Test的获取请求模板,如图2-14所示。

图2-14 获取请求模板的配置界面

要想使用获取请求模板,需要先给托管对象模型发送消息,告诉它将要使用的模板叫什么名字。发送完消息之后,就可以在返回的NSFetchRequest上面操作了。由于这种获取请求是根据模板创建出来的,所以开发者无需通过向其发送谓词来执行筛选操作。假如想修改这种获取请求(比方说,要对其进行排序),那么必须先制作它的一份拷贝,这是因为获取请求模板是根据不可变的模型(immutable model或unchangeable model)创建出来的。相关代码如程序清单2-8所示。

程序清单2-8 AppDelegate.m文件中的demo方法(演示获取请求模板的用法)

请按下列步骤修改Grocery Dude,以便使用名为Test的获取请求模板:

1.修改AppDelegate.m文件中的demo方法,用程序清单2-8中的代码替换原有的代码。程序清单2-8中的那行粗体代码是新加进来的,另外,修改过的代码中是没有谓词的。

运行应用程序,并观察控制台中的日志,你会发现这些托管对象均已按照名称排过序,而且只有名称中包含字母e的对象才会出现在控制台中,这些效果都是通过配置获取请求模板而实现出来的。正常的运行结果如图2-15所示。

图2-15 通过模板来筛选获取到的托管对象

不知你是否注意到了系统为这次fetch操作所生成的SQL语句。这条SQL语句的意思就是对获取到的各个Item进行筛选与排序(“获取”对应于语句中的“SELECT”;“筛选”对应于语句中的“WHERE”;“排序”对应于语句中的“ORDER BY”)。正常的运行效果如图2-16所示。

图2-16 系统自动生成的SQL语句