前言
NSPredicate 谓词, 其实 NSPredicate 有时候完全可以被 [for...][if..else] 替换来实现功能, 但会造成代码不够优雅, 一堆的判断循环 有时一句 NSPredicate 便可实现. 今天作者就聊一聊谓词的使用.
一 NSPredicate 理解说明
NSPredicate : 中文直译谓词, 用来定义逻辑条件约束的搜索 或 内存中的过滤。
- 如同语法中的谓词, 如 [3 大于 2]中"大于"就是一个就是谓词. 简单点说 它是逻辑判断, 如同过滤器, 筛选你所需要的.
- NSPredicate 与 [for...][if..else] 功能有相同的地方, 单纯比较在手机上运行效率, 作者查阅过[度娘] [谷哥], 只简单说会提升效率, 但未找到具体数据说明, 作者便不引用了.
- 按作者个人理解 NSPredicate 如同专车服务,抵达目的地下车; [for...][if..else]便是公交服务, 你自己判断在哪一站下车. 所以NSPredicate有一定的便捷性, [for...][if..else]有更广的适用性.
二 NSPredicate 语法说明
1
2
3
4
|
//一般初始化 NSPredicate *pred = [NSPredicate predicateWithFormat:@ "..." , ...]; //与具体对象 进行筛选判断, 返回为BOOL值 [pred evaluateWithObject:...] |
举例
1
2
3
4
5
6
|
//SELF支持小写, 代表正在被判断的对象自身 NSNumber *num = @999; NSPredicate *pred = [NSPredicate predicateWithFormat:@ "SELF = 999" ]; if ([pred evaluateWithObject:num]) { NSLog(@ "%@" , num); } |
//结果打印
999
一般很少这么用, 因为这么写都不如 [if..else] 便捷, 下面就讲一下 NSPredicate 的用法.
- 0. 集合类型方法说明
NSArray提供 : - (NSArray * )filteredArrayUsingPredicate:(NSPredicate *)predicate
NSMutableArray提供 : - (void)filterUsingPredicate:(NSPredicate *)predicate
NSSet提供 : - (NSSet *)filteredSetUsingPredicate:(NSPredicate *)predicate
NSMutableSet提供 : - (void)filterUsingPredicate:(NSPredicate *)predicate
- 1. 比较运算符 (以数组举例说明)
先封装创建 Person 类 (应该不用上代码)
1
2
3
4
5
6
|
//创建 Person 类数组 Person *p0 = [Person personName:@ "ZhangSan" withAge:20 withSex:@ "man" ]; Person *p1 = [Person personName:@ "HanMeiMei" withAge:12 withSex:@ "woman" ]; Person *p2 = [Person personName:@ "LiLei" withAge:13 withSex:@ "man" ]; Person *p3 = [Person personName:@ "XiaoHua" withAge:13 withSex:@ "woman" ]; NSArray *arr = @[p0, p1, p2, p3]; |
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
/** 比较运算符 */ NSPredicate *pred = [NSPredicate predicateWithFormat:@ "age < %@" , @20]; NSArray *resultArr = [arr filteredArrayUsingPredicate:pred]; NSLog(@ "年龄小于20 :%@" , resultArr); pred = [NSPredicate predicateWithFormat:@ "sex = 'woman' && age = 13" ]; resultArr = [arr filteredArrayUsingPredicate:pred]; NSLog(@ "年龄为13的女性 : %@" , resultArr); NSMutableArray *arrayM = [@[@20, @40, @50, @30, @60, @70] mutableCopy]; // 可以用 'BETWEEN {30, 50}' 代替 '>' // pred = [NSPredicate predicateWithFormat:@"SELF > 50"]; pred = [NSPredicate predicateWithFormat:@ "SELF BETWEEN {30, 50}" ]; [arrayM filterUsingPredicate:pred]; NSLog(@ "可变数组过滤 : %@" , arrayM); |
//结果打印
年龄小于20 :(
"HanMeiMei, 12, woman",
"LiLei, 13, man",
"XiaoHua, 13, woman"
)
年龄为13的女性 :(
"XiaoHua, 13, woman"
)可变数组过滤 : (
40,
50,
30
)
- 2. 字符串运算符
BEGINSWITH:检查某个字符串是否以指定的字符串开头(如判断字符串是否以a开头:BEGINSWITH 'a')
ENDSWITH:检查某个字符串是否以指定的字符串结尾
CONTAINS:检查某个字符串是否包含指定的字符串
LIKE:检查某个字符串是否匹配指定的字符串模板。其之后可以跟?代表一个字符和代表任意多个字符两个通配符。比如"name LIKE 'ac'",这表示name的值中包含ac则返回YES;"name LIKE '?ac'",表示name的第2、3个字符为ac时返回YES。
MATCHES:检查某个字符串是否匹配指定的正则表达式。虽然正则表达式的执行效率是最低的,但其功能是最强大的,也是我们最常用的。
注:字符串比较都是区分大小写和重音符号的。如:café和cafe是不一样的,Cafe和cafe也是不一样的。如果希望字符串比较运算不区分大小写和重音符号,请在这些运算符后使用[c],[d]选项。其中[c]是不区分大小写,[d]是不区分重音符号,其写在字符串比较运算符之后,比如:name LIKE[cd] 'cafe',那么不论name是cafe、Cafe还是café上面的表达式都会返回YES。
举例说明
1
2
3
4
5
6
|
//创建 Person 类数组 Person *p0 = [Person personName:@ "ZhangSan" withAge:20 withSex:@ "man" ]; Person *p1 = [Person personName:@ "HanMeiMei" withAge:12 withSex:@ "woman" ]; Person *p2 = [Person personName:@ "LiLei" withAge:13 withSex:@ "man" ]; Person *p3 = [Person personName:@ "XiaoHua" withAge:13 withSex:@ "woman" ]; NSArray *arr = @[p0, p1, p2, p3]; |
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
pred = [NSPredicate predicateWithFormat:@ "name LIKE '??an*'" ]; //名字中第三,四位是 an. resultArr = [arr filteredArrayUsingPredicate:pred]; NSLog(@ "1⃣️ : %@" , resultArr); pred = [NSPredicate predicateWithFormat:@ "name LIKE '*an*'" ]; //包含 字符串模板an. resultArr = [arr filteredArrayUsingPredicate:pred]; NSLog(@ "2⃣️ : %@" , resultArr); pred = [NSPredicate predicateWithFormat:@ "NOT (name CONTAINS 'ua')" ]; // 不包含 ua. resultArr = [arr filteredArrayUsingPredicate:pred]; NSLog(@ "3⃣️ : %@" , resultArr); pred = [NSPredicate predicateWithFormat:@ "name LIKE '*ua*'" ]; NSLog(@ "判断返回bool值 : %d" , [pred evaluateWithObject:p3]); |
//结果打印
"1⃣️ : (
"ZhangSan, 20, man"
)2⃣️ : (
"ZhangSan, 20, man",
"HanMeiMei, 12, woman"
)3⃣️ : (
"ZhangSan, 20, man",
"HanMeiMei, 12, woman",
"LiLei, 13, man"
)判断返回bool值 : 1
MATCHES 举例
1
2
3
|
/** 谓词匹配正则 */ NSString *phoneStr = @ "15242335566" ; NSLog(@ "验证 : %d" , [self checkPhoneNumber:phoneStr]); |
//结果打印
验证 : 1
1
2
3
4
5
6
7
8
9
|
- ( BOOL )checkPhoneNumber:(NSString *)phoneNumber { NSString *regex = @ "^[1][3-8]\\d{9}$" ; NSPredicate *pred = [NSPredicate predicateWithFormat:@ "SELF MATCHES %@" , regex]; return [pred evaluateWithObject:phoneNumber]; // 只有在正则表达式为^表达式$时才使用谓词,而不是所有情况都使用, 具体说明, 自行查看文章末尾参考链接, 作者便不再展开说明 } |
- 3. 集合运算符
ANY、SOME:集合中任意一个元素满足条件,就返回YES。
ALL:集合中所有元素都满足条件,才返回YES。
NONE:集合中没有任何元素满足条件就返回YES。如:NONE person.age < 18,表示person集合中所有元素的age>=18时,才返回YES。
IN:等价于SQL语句中的IN运算符,只有当左边表达式或值出现在右边的集合中才会返回YES。我们通过一个例子来看一下
举例
1
2
3
4
5
|
/** 从第二个数组中去除第一个数组中相同的元素 */ NSArray *filterArray = @[@ "ab" , @ "abc" ]; NSArray *array = @[@ "a" , @ "ab" , @ "abc" , @ "abcd" ]; pred = [NSPredicate predicateWithFormat:@ "NOT (SELF IN %@)" , filterArray]; NSLog(@ "去除相同元素 : %@" , [array filteredArrayUsingPredicate:pred]); |
//结果打印
去除相同元素 : (
a,
abcd
)
- 4. 谓词中使用占位符参数
首先如果我们想在谓词表达式中使用变量,那么我们需要了解下列两种占位符:
%K:用于动态传入属性名
%@:用于动态设置属性值
其实相当于变量名与变量值
> 除此之外,还可以在谓词表达式中使用动态改变的属性值,就像环境变量一样
1
|
NSPredicate *pred = [NSPredicate predicateWithFormat:@ "SELF CONTAINS $VALUE" ]; |
> 上述表达式中,$VALUE是一个可以动态变化的值,它其实最后是在字典中的一个key,所以可以根据你的需要写不同的值,但是必须有$开头,随着程序改变$VALUE这个谓词表达式的比较条件就可以动态改变。
举例
1
2
3
4
5
6
|
//创建 Person 类数组 Person *p0 = [Person personName:@ "ZhangSan" withAge:20 withSex:@ "man" ]; Person *p1 = [Person personName:@ "HanMeiMei" withAge:12 withSex:@ "woman" ]; Person *p2 = [Person personName:@ "LiLei" withAge:13 withSex:@ "man" ]; Person *p3 = [Person personName:@ "XiaoHua" withAge:13 withSex:@ "woman" ]; NSArray *arr = @[p0, p1, p2, p3]; |
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
NSString *property = @ "name" ; NSString *value = @ "LiLei" ; // 该谓词的作用是如果元素中property属性含有值value时就取出放入新的数组内,这里是name包含LiLei pred = [NSPredicate predicateWithFormat:@ "%K CONTAINS %@" , property, value]; NSArray *newArray = [arr filteredArrayUsingPredicate:pred]; NSLog(@ "newArray:%@" , newArray); // 创建谓词,属性名改为age,要求这个age包含$VALUE字符串 NSPredicate *predTemp = [NSPredicate predicateWithFormat:@ "%K > $VALUE" , @ "age" ]; // 指定$SUBSTR的值为 12 这里注释中的$SUBSTR改为$VALUE NSPredicate *pred1 = [predTemp predicateWithSubstitutionVariables:@{@ "VALUE" : @12}]; NSArray *newArray1 = [arr filteredArrayUsingPredicate:pred1]; NSLog(@ "newArray1:%@" , newArray1); // 修改 $SUBSTR的值为13, 这里注释中的SUBSTR改为$VALUE NSPredicate *pred2 = [predTemp predicateWithSubstitutionVariables:@{@ "VALUE" : @13}]; NSArray *newArray2 = [arr filteredArrayUsingPredicate:pred2]; NSLog(@ "newArray2:%@" , newArray2); |
//结果打印
newArray:(
"LiLei, 13, man"
)newArray1:(
"ZhangSan, 20, man",
"LiLei, 13, man",
"XiaoHua, 13, woman"
)newArray2:(
"ZhangSan, 20, man"
)
- 5. 谓词中保留字
保留字
下列单词都是保留字(不论大小写)
AND、OR、IN、NOT、ALL、ANY、SOME、NONE、LIKE、CASEINSENSITIVE、CI、MATCHES、CONTAINS、BEGINSWITH、ENDSWITH、BETWEEN、NULL、NIL、SELF、TRUE、YES、FALSE、NO、FIRST、LAST、SIZE、ANYKEY、SUBQUERY、CAST、TRUEPREDICATE、FALSEPREDICATE
注:虽然大小写都可以,但是更推荐使用大写来表示这些保留字
参考 : https://www.zzvips.com/article/164588.html
以上 !
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对服务器之家的支持。
原文链接:https://www.jianshu.com/p/d4098bc9488d