iOS中NSPredicate谓词的使用

前言

NSPredicate 谓词,其实 NSPredicate 有时候完全可以被 [for...][if..else] 替换来实现功能,但会造成代码不够优雅,一堆的判断循环 有时一句 NSPredicate 便可实现. 今天作者就聊一聊谓词的使用.

一 NSPredicate 理解说明

NSPredicate : 中文直译谓词,用来定义逻辑条件约束的搜索 或 内存中的过滤。

  • 如同语法中的谓词,如 [3 大于 2]中"大于"就是一个就是谓词. 简单点说 它是逻辑判断,如同过滤器,筛选你所需要的.
  • NSPredicate 与 [for...][if..else] 功能有相同的地方,单纯比较在手机上运行效率,作者查阅过[度娘] [谷哥],只简单说会提升效率,但未找到具体数据说明,作者便不引用了.
  • 按作者个人理解 NSPredicate 如同专车服务,抵达目的地下车; [for...][if..else]便是公交服务,你自己判断在哪一站下车. 所以NSPredicate有一定的便捷性,[for...][if..else]有更广的适用性.

二 NSPredicate 语法说明

//一般初始化
 NSPredicate *pred = [NSPredicate predicateWithFormat:@"...",...];
//与具体对象 进行筛选判断,返回为BOOL值
[pred evaluateWithObject:...]

举例

//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 类 (应该不用上代码)

//创建 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];
/** 比较运算符 */
 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,woman"
)

年龄为13的女性 :(
   "XiaoHua,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。

举例说明

//创建 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,p3];
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,
   "HanMeiMei,woman"
)

3⃣️ : (
    "ZhangSan,
    "HanMeiMei,man"
)

判断返回bool值 : 1

MATCHES 举例

 /** 谓词匹配正则 */
 NSString *phoneStr = @"15242335566";
 NSLog(@"验证 : %d",[self checkPhoneNumber:phoneStr]);

//结果打印
验证 : 1

- (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。我们通过一个例子来看一下

举例

/** 从第二个数组中去除第一个数组中相同的元素 */
 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:用于动态传入属性名
%@:用于动态设置属性值
其实相当于变量名与变量值

> 除此之外,还可以在谓词表达式中使用动态改变的属性值,就像环境变量一样

NSPredicate *pred = [NSPredicate predicateWithFormat:@"SELF CONTAINS $VALUE"];

> 上述表达式中,$VALUE是一个可以动态变化的值,它其实最后是在字典中的一个key,所以可以根据你的需要写不同的值,但是必须有$开头,随着程序改变$VALUE这个谓词表达式的比较条件就可以动态改变。

举例

//创建 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,p3];
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,man"
)

newArray1:(
    "ZhangSan,woman"
)

newArray2:(
    "ZhangSan,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.jb51.net/article/155027.htm

以上 !

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对我们的支持。

相关文章

当我们远离最新的 iOS 16 更新版本时,我们听到了困扰 Apple...
欧版/美版 特别说一下,美版选错了 可能会永久丧失4G,不过只...
一般在接外包的时候, 通常第三方需要安装你的app进行测...
前言为了让更多的人永远记住12月13日,各大厂都在这一天将应...