iOS Foundation框架之谓词NSPredicate
很多初学者都很少接触NSPredicate,即使接触了NSPredicate 也不太了解NSPredicate的使用场景。此处根据我的使用,简单介绍一下NSPredicate的使用场景:
NSPredicate 主要是用来查询、条件过滤;
最常用的场景就是在自定义的数据模型对象
中根据条件来查询相关信息,例如在手机通讯录中根据个人信息的Model所包含的name属性,来进行搜索
** 简言之** NSPredicate可以判断某个对象的某一个属性是否符合某一条件
1. NSPredict简单说明
- NSPredict 谓词可以通过定义一个逻辑条件,来搜索查询、过滤信息
- NSPredict主要包含三个子类:NSComparisonPredicate、NSCompoundPredicate、NSExpression
2. NSPredict 表达式
在介绍NSPredict的使用之前,我们必须先要了解如何正确的书写谓词表达式
-
比较运算符
-
= 、 == :判断两个表达式是否相等
-
>= 、=>: 左侧表达式是否大于或等于右侧表达式
-
<= 、 =< :左侧表达式是否小于或等于右侧表达式
-
> 、< :左侧表达式是否大于、小于右侧表达式
-
!= 、<>: 两个表达式是否不相等
-
BETWEEN :”表达式 BEYWEEN {最小值,最大值}“ ,表达式必须大于等于最小值或小于等于最大值
-
逻辑运算符
-
AND、&& :两个表达式都为真是,结果为真;有假则结果为假
-
OR、|| :两个表达式有一个结果为真,结果为真;同为假,则结果为假
-
NOT 、!: 表达式结果取反
-
字符串
比较运算符 -
BEGINSWITH :字符串是否以某一子字符串开头
-
ENDSWITH : 字符串是否以某一子字符串结尾
-
CONTAINS :字符串是否包含某一子字符串
-
LIKE :字符串是否匹配指定的字符串
模板
- 模板中可以包含通配符: *(任意多个字符)、?(一个字符)
- title LIKE *abc? :title现有任意多的字符串,结尾必须为abc+任意一个字符
-
MATCHES : 字符串是否匹配指定的正则表达式;正则表达式功能强大,但是执行效率低。能用谓词表达式的就不要用正则表达式
注意:字符串比较运算符默认区分大小写和重音符号
1.如果希望字符串比较运算符不区分大小写,可以再运算符后添加[c]
2.如果希望字符串比较运算符不区分重音符号,可以再运算符后添加[d]
一般都是在运算符后面添加[cd],代表不区分大小写和重音符号
-
集合操作相关的运算符
-
ANY 、SOME :集合中任意一个元素满足条件,返回YES
-
ALL :集合中多有元素满足条件,返回YES
-
NONE :集合中任何元素不满足条件,返回YES
-
IN :左边的表达式(值) 在右边的集合中存在,返回YES
-
array[index] : 返回数组index索引处的元素
-
array[FIRST] : 返回数组第一个元素
-
array[LAST] : 返回数组最后一个元素
-
array[SIZE] : 返回数组元素个数
-
谓词表达式中的直接量
-
FALSE、NO : 逻辑假
-
TRUE、YES : 逻辑真
-
NULL、NIL : 空值
-
SELF :被判断的对象本身
-
"text"('text') : 字符串
-
数组:数组元素以英文逗号隔开
-
数值直接量:整数、小数、科学技术法
-
各进制数:0x(十六进制)、0o(八进制)、0b(二进制)
注意:谓词表达式中" "与' '效果相同,但是“ ”与‘ ’应该匹配 -
保留字:大写单词是保留字
MATCHES、CONTAINS、BEGINSWITH、ENDSWIHT、BETWEEN、NULL、NIL、SELF、AND、OR、IN、NOT、ALL、ANY、SOME、NONE、LIKE、CASEINSENSITIVE、CI、TRUE、YES、FALSE、NO、FIRST、LAST、SIZE、ANYKEY、SUBQUERY、CAST、TRUEPREDICATE、FALSEOREDICATE
2. NSPredict定义
//以一个NSPredict字符串来创建一个NSPredict对象
+ (NSPredicate *)predicateWithFormat:(NSString *)predicateFormat, ...;
+ (NSPredicate *)predicateWithFormat:(NSString *)predicateFormat argumentArray:(nullable NSArray *)arguments;
+ (NSPredicate *)predicateWithFormat:(NSString *)predicateFormat arguments:(va_list)argList;
//通过一个对象来计算谓词的结果 真/假
- (BOOL)evaluateWithObject:(nullable id)object;
- 示例代码
ZZYPerson * person1 = [[ZZYPerson alloc]initWithName:@"zhang san" Age:@"21"];
NSPredicate * pred1 = [NSPredicate predicateWithFormat:@"name CONTAINS[cd] %@",@"san"];
NSPredicate * pred2 = [NSPredicate predicateWithFormat:@"name CONTAINS[cd] 'san'"];
//注意第二种写法要将字符串用单引号包起来
BOOL isHave = [pred1 evaluateWithObject:person1];
NSLog(@"%d",isHave); //输出值为:1
NSLog(@"%@---%@",person1.name,person1.age);//输出值为:zhang san---21
** 注意 **上述代码中pred2中的写法,字符串要用单引号括起来(比如sql语句中的字符串也要包含起来)
3. NSPredict 过滤查找集合元素
- NSPredict 本质上就是一个逻辑条件,NSPredict 运算的结果就是一个BOOL值
- NSPredict 一个比较常用的功能就是
对集合元素的过滤
; 自动遍历集合元素------>根据元素判断 NSPredict 的结果------>结果为YES时,集合元素保存 - 注意谓词过滤不可变集合,结果返回符合条件的新集合;谓词过滤可变集合,直接将集合中不符合条件的元素去掉
@interface NSArray<ObjectType> (NSPredicateSupport)
//根据谓词条件过滤数组,返回符合条件的元素组成的新数组
- (NSArray<ObjectType> *)filteredArrayUsingPredicate:(NSPredicate *)predicate; // evaluate a predicate against an array of objects and return a filtered array
@end
@interface NSMutableArray<ObjectType> (NSPredicateSupport)
//根据谓词条件过滤数组,去掉数组中不符合条件的的元素
- (void)filterUsingPredicate:(NSPredicate *)predicate; // evaluate a predicate against an array of objects and filter the mutable array directly
@end
@interface NSSet<ObjectType> (NSPredicateSupport)
////根据谓词条件过滤集合,返回符合条件的元素组成的新集合
- (NSSet<ObjectType> *)filteredSetUsingPredicate:(NSPredicate *)predicate NS_AVAILABLE(10_5, 3_0); // evaluate a predicate against a set of objects and return a filtered set
@end
@interface NSMutableSet<ObjectType> (NSPredicateSupport)
//根据谓词条件过滤集合,去掉集合中不符合条件的的元素
- (void)filterUsingPredicate:(NSPredicate *)predicate NS_AVAILABLE(10_5, 3_0); // evaluate a predicate against a set of objects and filter the mutable set directly
@end
- 示例代码
NSArray * array = @[@"libai",@"dufu",@"sushi",@"dumu"];
NSPredicate * pred = [NSPredicate predicateWithFormat:@"SELF like %@",@"du*"];
NSArray * resultArr = [array filteredArrayUsingPredicate:pred];
NSLog(@"%@",resultArr);//输出值:(
dufu,
dumu
)
NSSet * set = [NSSet setWithObjects:
[[ZZYPerson alloc]initWithName:@"li si" Age:@"25"],
[[ZZYPerson alloc]initWithName:@"zhang san" Age:@"20"],
[[ZZYPerson alloc]initWithName:@"wang wu" Age:@"18"],nil
];
NSPredicate * pred3 = [NSPredicate predicateWithFormat:@"name CONTAINS 'ang'"];
NSSet * resultSet = [set filteredSetUsingPredicate:pred3];
for (ZZYPerson * person in resultSet) {
NSLog(@"%@",person.name);
}
NSArray * array3 = @[[[ZZYPerson alloc]initWithName:@"li si" Age:@"25"],
[[ZZYPerson alloc]initWithName:@"zhang san" Age:@"20"],
[[ZZYPerson alloc]initWithName:@"wang wu" Age:@"18"]];
NSArray * array4 = [array3 filteredArrayUsingPredicate:pred3];
for (ZZYPerson * person in array4) {
NSLog(@"%@",person.name);
}
//输出值:
zhang san
wang wu
4. NSPredict 的占位符参数
- 通过使用占位符,在谓词表达式中使用变量
- %K : 动态传入属性名
- %@ : 动态设置属性值
- $SUBSTR : 一个动态变化的值,可以通过它动态改变比较条件
//设置NSPredict 中的可变参数,并计算结果
- (BOOL)evaluateWithObject:(nullable id)object substitutionVariables:(nullable NSDictionary<NSString *, id> *)bindings NS_AVAILABLE(10_5, 3_0); // single pass evaluation substituting variables from the bindings dictionary for any variable expressions encountered
//设置NSPredict 的可变参数,返回一个NSPredict 对象
- (instancetype)predicateWithSubstitutionVariables:(NSDictionary<NSString *, id> *)variables; // substitute constant values for variables
- 示例代码
ZZYPerson * person1 = [[ZZYPerson alloc]initWithName:@"zhang san" Age:@"21"];
ZZYPerson * person2 = [[ZZYPerson alloc]initWithName:@"li si" Age:@"25"];
ZZYPerson * person3 = [[ZZYPerson alloc]initWithName:@"stark" Age:@"11"];
ZZYPerson * person4 = [[ZZYPerson alloc]initWithName:@"sunny" Age:@"30"];
NSArray * array2 = @[person1,person2,person3,person4];
NSString * name = @"age";
NSString * age = @"3";
NSPredicate * changePre1 = [NSPredicate predicateWithFormat:@"%K CONTAINS[cd] %@",name,age];
NSArray * newArray2 = [array2 filteredArrayUsingPredicate:changePre1];
for (ZZYPerson * person in newArray2) {
NSLog(@"newArray2%@----%@",person.name,person.age);
}
//name中包含$SUBSTR的字串
NSPredicate * changePre2 = [NSPredicate predicateWithFormat:@"%K CONTAINS[cd] $SUBSTR",@"name"];
//指定$SUBSTR的值为sun
NSPredicate * newChangePre2 = [changePre2 predicateWithSubstitutionVariables:@{@"SUBSTR":@"sun"}];
NSArray * newArray3 = [array2 filteredArrayUsingPredicate:newChangePre2];
for (ZZYPerson * person in newArray3) {
NSLog(@"newArray3%@----%@",person.name,person.age);
}
NSPredicate * newChangePre3 = [changePre2 predicateWithSubstitutionVariables:[NSDictionary dictionaryWithObjectsAndKeys:@"ang",@"SUBSTR", nil]];
NSArray * newArray4 = [array2 filteredArrayUsingPredicate:newChangePre3];
for (ZZYPerson * person in newArray4) {
NSLog(@"newArray4%@----%@",person.name,person.age);
}
//输出结果:
2016-06-27 23:07:11.820 ZZYNSPredicate[4540:96427] newArray2sunny----30
2016-06-27 23:07:11.820 ZZYNSPredicate[4540:96427] newArray3sunny----30
2016-06-27 23:07:11.820 ZZYNSPredicate[4540:96427] newArray4zhang san----21