欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  移动技术

iOS中NSPredicate谓词的使用

程序员文章站 2023-11-22 11:05:16
前言 nspredicate 谓词, 其实 nspredicate 有时候完全可以被 [for...][if..else] 替换来实现功能, 但会造成代码不够优雅,...

前言

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, 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。

举例说明

//创建 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];
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 举例

 /** 谓词匹配正则 */
 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, p1, p2, 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, 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

注:虽然大小写都可以,但是更推荐使用大写来表示这些保留字

参考 :

以上 !

总结

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