CoreData—使用进阶
NSPredicate
概述
在iOS开发过程中,很多需求都需要用到过滤条件。例如过滤一个集合对象中存储的对象,可以通过Foundation框架下的NSPredicate类来执行这个操作。
CoreData中可以通过设置NSFetchRequest类的predicate属性,来设置一个NSPredicate类型的谓词对象当做过滤条件。通过设置这个过滤条件,可以只获取符合过滤条件的托管对象,不会将所有托管对象都加载到内存中。这样是非常节省内存和加快查找速度的,设计一个好的NSPredicate可以优化CoreData搜索性能。
语法
NSPredicate更加偏向于自然语言,不像SQLite一样有很多固定的语法,看起来也更加清晰易懂。例如下面需要查找条件为年龄30岁以上,并且包括30岁的条件。
1 |
[NSPredicate predicateWithFormat:@ "age >= 30" ]
|
过滤集合对象
可以通过NSPredicate对iOS中的集合对象执行过滤操作,可以是NSArray、NSSet及其子类。
对不可变数组NSArray执行的过滤,过滤后会返回一个NSArray类型的结果数组,其中存储着符合过滤条件的对象。
1 |
NSArray *results = [array filteredArrayUsingPredicate:predicate]
|
对可变数组NSMutableArray执行的过滤条件,过滤后会直接改变原集合对象内部存储的对象,删除不符合条件的对象。
1 |
[arrayM filterUsingPredicate:predicate]
|
复合过滤条件
谓词不只可以过滤简单条件,还可以过滤复杂条件,设置复合过滤条件。
1 |
[NSPredicate predicateWithFormat:@ "(age < 25) AND (firstName = XiaoZhuang)" ]
|
当然也可以通过NSCompoundPredicate对象来设置复合过滤条件,返回结果是一个NSPredicate的子类NSCompoundPredicate对象。
1 |
[[NSCompoundPredicate alloc] initWithType:NSAndPredicateType subpredicates:@[predicate1, predicate2]]
|
枚举值NSCompoundPredicateType参数,可以设置三种复合条件,枚举值非常直观很容易看懂。
-
NSNotPredicateType
-
NSAndPredicateType
-
NSOrPredicateType
基础语法
下面是列举的一些NSPredicate的基础语法,这些语法看起来非常容易理解,更复杂的用法可以去看苹果的官方API。
正则表达式
NSPredicate中还可以使用正则表达式,可以通过正则表达式完成一些复杂需求,这使得谓词的功能更加强大,例如下面是一个手机号验证的正则表达式。
1 2 |
NSString *mobile = @ "^1(3[0-9]|5[0-35-9]|8[025-9])\d{8}$" ;
NSPredicate *regexmobile = [NSPredicate predicateWithFormat:@ "SELF MATCHES %@" , mobile];
|
模糊查询
NSPredicate支持对数据的模糊查询,例如下面使用通配符来匹配包含lxz的结果,具体CoreData中的使用在下面会讲到。
1 |
[NSPredicate predicateWithFormat:@ "name LIKE %@" , @ "*lxz*" ]
|
keyPath
NSPredicate在创建查询条件时,还支持设置被匹配目标的keyPath,也就是设置更深层被匹配的目标。例如下面设置employee的name属性为查找条件,就是用点语法设置的keyPath。
1 |
[NSPredicate predicateWithFormat:@ "employee.name = %@" , @ "lxz" ]
|
设置查询条件
在之前的文章中,执行下面MOC的fetchRequest方法,一般都需要传入一个NSFetchRequest类型的参数。这个request参数可以做一些设置操作,这样就可以以较优的性能获取指定的数据。
1 |
- (nullable NSArray *)executeFetchRequest:(NSFetchRequest *)request error:(NSError **)error;
|
NSFetchRequest
在执行fetch操作前,可以给NSFetchRequest设置一些参数,这些参数包括谓词、排序等条件,下面是一些基础的设置。
-
设置查找哪个实体,从数据库的角度来看就是查找哪张表,通过fetchRequestWithEntityName:或初始化方法来指定表名。
-
通过NSPredicate类型的属性,可以设置查找条件,这个属性在开发中用得最多。NSPredicate可以包括固定格式的条件以及正则表达式。
-
通过sortDescriptors属性,可以设置获取结果数组的排序方式,这个属性是一个数组类型,也就是可以设置多种排序条件。(但是注意条件不要冲突)
-
通过fetchOffset属性设置从查询结果的第几个开始获取,通过fetchLimit属性设置每次获取多少个。主要用于分页查询,后面会讲。
MOC执行fetch操作后,获取的结果是以数组的形式存储的,数组中存储的就是托管对象。NSFetchRequest提供了参数resultType,参数类型是一个枚举类型。通过这个参数,可以设置执行fetch操作后返回的数据类型。
-
NSManagedObjectResultType: 返回值是NSManagedObject的子类,也就是托管对象,这是默认选项
-
NSManagedObjectIDResultType: 返回NSManagedObjectID类型的对象,也就是NSManagedObject的ID,对内存占用比较小。MOC可以通过NSManagedObjectID对象获取对应的托管对象,并且可以通过缓存NSManagedObjectID参数来节省内存消耗
-
NSDictionaryResultType: 返回字典类型对象
-
NSCountResultType: 返回请求结果的count值,这个操作是发生在数据库层级的,并不需要将数据加载到内存中
设置获取条件
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 |
// 建立获取数据的请求对象,并指明操作Employee表
NSFetchRequest *request = [NSFetchRequest fetchRequestWithEntityName:@ "Employee" ];
// 设置请求条件,通过设置的条件,来过滤出需要的数据
NSPredicate *predicate = [NSPredicate predicateWithFormat:@ "name = %@" , @ "lxz" ];
request.predicate = predicate;
// 设置请求结果排序方式,可以设置一个或一组排序方式,最后将所有的排序方式添加到排序数组中
NSSortDescriptor *sort = [NSSortDescriptor sortDescriptorWithKey:@ "height" ascending:YES];
// NSSortDescriptor的操作都是在SQLite层级完成的,不会将对象加载到内存中,所以对内存的消耗是非常小的
|