谓词: 简单的说就是一个过滤器, 符合条件的留下, 不符合条件的删除
一. NSPredicate的基本语法
只要使用谓词(NSPredicate)都需要为谓词定义谓词表达式, 而这个表达式必须是一个返回BOOL的值
谓词表达式由表达式, 运算符和值构成
1: 比较运算符
(1)=、==:判断两个表达式是否相同
备注: 在谓词中=和==是都是相同的意思, 不是赋值
(2)>=, =>: 判断左边的表达式的值是否 大于或等于 右边表达式的值
(3)<=, =<: 判断左边的表达式的值是否 小于或等于 右边表达式的值
(4)>: 判断左边表达式的值是否 大于 右边表达式的值
(5)<: 判断左边表达式的值是否 小于 右边表达式的值
(6)!=, <>: 判断两个表达式是否 不相等
(7)BETWEEN: 该表达式必须满足: 表达式 BETWEEN {下限,上限} 的格式, 要求该表达式必须 大于或等于下限, 并小于等于上限
例:
NSNumber *testNumber = @16;
NSPredicate *predicate = [NSPredicate predicateWithFormat:@"SELF BETWEEN {12,19}"];
if ([predicate evaluateWithObject:testNumber]) {
NSLog(@"testNumber = %@", testNumber);
}
else {
NSLog(@"不符合");
}
2: 逻辑运算符:
(1)AND, &&: 逻辑与, 要求两个表达式的值都为YES时, 结果才为YES
例: NSArray *testArray = @[@1, @2, @3, @4, @5, @6];
NSPredicate *predicate = [NSPredicate predicateWithFormat:@"SELF > 2 && SELF < 5"];
NSArray *filterArray = [testArray filteredArrayUsingPredicate:predicate];
NSLog(@"filterArray: %@",filterArray);
输出结果:filterArray: (
3,
4
)
(2)OR, ||: 逻辑或, 要求其中一个表达式为YES时, 结果就是YES
例: NSArray *testArray = @[@1, @2, @3, @4, @5, @6];
NSPredicate *predicate = [NSPredicate predicateWithFormat:@"SELF < 2 || SELF > 5"];
NSArray *filterArray = [testArray filteredArrayUsingPredicate:predicate];
NSLog(@"filterArray: %@",filterArray);
输出结果:filterArray: (
1,
6
)
(3)NOT, !: 逻辑非, 对原有的表达式取反
例: NSNumber *testNumber = @1;
NSPredicate *predicate = [NSPredicate predicateWithFormat:@"SELF != 1"];
if ([predicate evaluateWithObject:testNumber]) {
NSLog(@"testNumber = %@", testNumber);
}
else {
NSLog(@"不符合条件");
}
输出结果: 不符合条件
3: 字符串比较运算符:
(1)BEGINSWITH: 检查某个字符串是否已指定的字符串开头
例: NSString *testString = @"testString";
NSPredicate *predicate = [NSPredicate predicateWithFormat:@"SELF BEGINSWITH 't'"];
if ([predicate evaluateWithObject:testString]) {
NSLog(@"testString = %@", testString);
}
else {
NSLog(@"不符合条件");
}
输出结果: testString = testString
(2)ENDSWITH:检查某个字符串是否以指定的字符串结尾
(3)CONTAINS: 检查某个字符串是否包含指定的字符串
例: NSString *testString = @"testString";
NSPredicate *predicate = [NSPredicate predicateWithFormat:@"SELF CONTAINS 'ts'"];
if ([predicate evaluateWithObject:testString]) {
NSLog(@"testString = %@", testString);
}
else {
NSLog(@"不符合条件");
}
输出结果: 不符合条件
备注: ts要用单引号' ' 括住, 否则会崩溃, 切记;
(4) LIKE: 检查某个字符串是否匹配指定的字符串模板. 其之后可以跟 ? 代表一个字符 和 * 代表任意多个字符 两个通配符;
例: "testString LIKE '*tS*'": 表示如果testString的值中包含ac则返回YES, 此时和CONTAINS相同
NSString *testString = @"testString";
NSPredicate *predicate = [NSPredicate predicateWithFormat:@"SELF LIKE '*tS*'"];
if ([predicate evaluateWithObject:testString]) {
NSLog(@"testString = %@", testString);
}
else {
NSLog(@"不符合条件");
}
输出结果:testString = testString
例: "testString LIKE '?tS*", 表示testString的第2, 3个字符为ac时, 返回YES
NSString *testString = @"testString";
NSPredicate *predicate = [NSPredicate predicateWithFormat:@"SELF LIKE '?tS*'"];
if ([predicate evaluateWithObject:testString]) {
NSLog(@"testString = %@", testString);
}
else {
NSLog(@"不符合条件");
}
输出结果: 不符合条件
(5)MATCHES: 检查某个字符串是否匹配指定的正则表达式。虽然正则表达式的执行效率是最低的, 但其功能是最强大的, 也是我们 最常用 的。
注: 字符串比较都是区分 大小写 和重音符号 的。如:café和cafe是不一样的,Cafe和cafe也是不一样的。如果希望字符串比较运算不区分大小写和重音符号, 请在这些运算符后使用 [c], [d]选项。其中[c] 是不区分大小写,[d]是不区分重音符号,其写在字符串比较运算符之后,比如: testString LIKE[cd] 'cafe', 那么不论testString是cafe,Cafe还是café上面的表达式都会返回YES。
4. 集合运算符
(1)ANY、SOME: 集合中任意一个元素满足条件,就返回YES。
NSArray *testArray = @[@"zhao",@"qian",@"sun"];
NSPredicate *predicate = [NSPredicate predicateWithFormat:@"ANY SELF BEGINSWITH 'su'"];
if ([predicate evaluateWithObject:testArray]) {
NSLog(@"testArray = %@", testArray);
}
else {
NSLog(@"不符合条件");
}
输出结果:testArray = (
zhao,
qian,
sun
)
(2)ALL: 集合中所有元素都满足条件,才返回YES
NSArray *testArray = @[@"zhao",@"qian",@"sun"];
NSPredicate *predicate = [NSPredicate predicateWithFormat:@"ALL SELF BEGINSWITH 'su'"];
if ([predicate evaluateWithObject:testArray]) {
NSLog(@"testArray = %@", testArray);
}
else {
NSLog(@"不符合条件");
}
输出结果: 不符合条件
(3)NONE: 集合中没有任何元素满足条件就返回YES
NSArray *testArray = @[@4, @5, @6, @7];
NSPredicate *predicate = [NSPredicate predicateWithFormat:@"NONE SELF < 3"];
if ([predicate evaluateWithObject:testArray]) {
NSLog(@"testArray = %@", testArray);
}
else {
NSLog(@"不符合条件");
}
输出结果:testArray = (
4,
5,
6,
7
)
(4)IN: 等价于SQL语句中的IN运算符, 只有当左边表达式或值出现在右边的集合中才会返回YES
NSArray *filterArray = @[@"ab", @"abc"];
NSArray *testArray = @[@"a", @"ab", @"abc", @"abcd"];
NSPredicate *predicate = [NSPredicate predicateWithFormat:@"NOT (SELF IN %@)", filterArray];
NSLog(@"%@",[testArray filteredArrayUsingPredicate:predicate]);
输出结果:(
a,
abcd
)
上面那段代码的作用是将testArray中和filterArray中相同的元素去除
5.直接量
在谓词表达式中可以使用如下直接量:
(1)FALSE、NO:代表逻辑假
(2)TRUE、YES:代表逻辑真
(3)NULL、NIL:代表空值
(4)SELF:代表正在被判断的对象自身
(5)"string"或'string':代表字符串
(6)数组:和c中的写法相同, 如:{'one','two','three'}
(7)数值: 包括证书、小数和科学计数法表示的形式
(8)十六进制数: 0x开头的数字
(9)八进制: 0o开头的数字
(10)二进制:0b开头的数字
6.保留字
下列单词都是保留字(不论大小写)
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
注:虽然大小写都可以,但是更推荐使用大写来表示这些保留字
二、谓词的用法
1.定义谓词
NSPredicate *predicate = [NSPredicate predicateWithFormat:@""];
例一:
定义模型:
ZLPersonModel.h
#import <Foundation/Foundation.h>
typedef NS_ENUM(NSInteger, ZLPersonSex){
ZLPersonSexMale = 0,
ZLPersonSexFemale
};
@interface ZLPersonModel : NSObject
/** 姓名 */
@property (nonatomic, copy) NSString *name;
/** 年龄 */
@property (nonatomic, assign, readonly) NSUInteger age;
/** 性别 */
@property (nonatomic, assign, readonly) ZLPersonSex sex;
+ (instancetype)personWithName:(NSString *)name age:(NSUInteger)age sex:(ZLPersonSex)sex;
@end
ZLPersonModel.m
#import "ZLPersonModel.h"
@implementation ZLPersonModel
- (instancetype)initWithName:(NSString *)name age:(NSUInteger)age sex:(ZLPersonSex)sex {
self = [super init];
if (self) {
_name = name;
_age = age;
_sex = sex;
}
return self;
}
+ (instancetype)personWithName:(NSString *)name age:(NSUInteger)age sex:(ZLPersonSex)sex {
return [[self alloc] initWithName:name age:age sex:sex];
}
@end
开始使用:
ZLPersonModel *sunnyzl = [ZLPersonModel personWithName:@"sunnyzl" age:29 sex:ZLPersonSexMale];
ZLPersonModel *jack = [ZLPersonModel personWithName:@"jack" age:22 sex:ZLPersonSexMale];
//1: 判断姓名是否是以s开头
NSPredicate *pred1 = [NSPredicate predicateWithFormat:@"name LIKE 's*'"];
//输出结果: sunnyzl: 1, jack: 0
NSLog(@"sunnyzl: %d, jack: %d",[pred1 evaluateWithObject:sunnyzl], [pred1 evaluateWithObject:jack]);
//2: 判断年龄是否大于25
NSPredicate *pred2 = [NSPredicate predicateWithFormat:@"age > 25"];
//输出结果: sunnyzl的年龄是否大于25: 1, jack的年龄是否大于25: 0
NSLog(@"sunnyzl的年龄是否大于25: %d, jack的年龄是否大于25: %d",[pred2 evaluateWithObject:sunnyzl], [pred2 evaluateWithObject:jack]);
2.使用谓词过滤集合(很重要)
其实 谓词 本身就代表了一个逻辑条件,计算谓词之后返回的结果永远为BOOL类型的值。而谓词最常用的功能就是对集合进行过滤。当程序使用谓词对集合元素进行过滤时,程序会自动遍历其元素,并根据集合元素来计算谓词的值,当这个集合中的元素计算谓词并返回YES时,这个元素才会被保留下来。请注意 程序会自动遍历其元素,它会将自动遍历过之后返回为YES的值重新组合成一个集合返回。
//NSArray提供了如下方法使用谓词来过滤集合
//根据指定的谓词过滤集合,并将符合条件的元素组成新的集合返回(应用于NSArray中)
//参数: 指定的谓词
//返回:符合谓词条件的元素组成的集合
- (NSArray<ObjectType> *)filteredArrayUsingPredicate:(NSPredicate *)predicate;
//NSMutableArray提供了如下方法使用谓词来过滤集合
//根据指定的谓词过滤集合,剔除集合中不符合条件的元素
//参数: 指定的谓词
- (void)filterUsingPredicate:(NSPredicate *)predicate;
//NSSet提供了如下方法使用谓词来过滤集合
//作用同NSArray中的方法
- (NSSet<ObjectType> *)filteredSetUsingPredicate:(NSPredicate *)predicate;
//NSMutableSet提供了如下方法使用谓词来过滤集合
//作用同NSMutableArray中的方法
- (void)filterUsingPredicate:(NSPredicate *)predicate API_AVAILABLE(macos(10.5), ios(3.0), watchos(2.0), tvos(9.0));
备注: 通过上面的描述可以看出, 使用谓词过滤不可变集合和可变集合的区别是: 过滤不可变集合时, 会 返回符合条件 的集合元素组成的新集合; 过滤可变集合时, 没有返回值, 会 直接剔除不符合条件 的集合元素.
例一:
NSMutableArray *arrayM = [@[@20, @40, @50, @30, @60, @70] mutableCopy];
//过滤大于50的值
NSPredicate *pred1 = [NSPredicate predicateWithFormat:@"SELF > 50"];
[arrayM filterUsingPredicate:pred1];
NSLog(@"arrayM = %@",arrayM);
NSArray *array = @[[ZLPersonModel personWithName:@"Jack" age:20 sex:ZLPersonSexMale],
[ZLPersonModel personWithName:@"Rose" age:22 sex:ZLPersonSexFemale],
[ZLPersonModel personWithName:@"Jackson" age:30 sex:ZLPersonSexMale],
[ZLPersonModel personWithName:@"Johnson" age:35 sex:ZLPersonSexMale]];
//取出名字中包含'son'的元素
NSPredicate *pred2 = [NSPredicate predicateWithFormat:@"name CONTAINS 'son'"];
NSArray *newArray = [array filteredArrayUsingPredicate:pred2];
NSLog(@"newArray = %@",newArray);
输出结果:arrayM = (
60,
70
)
newArray = (
"[name = Jackson, age = 30, sex = ZLPersonSexMale]",
"[name = Johnson, age = 35, sex = ZLPersonSexMale]"
)
3. 在谓词中使用占位符参数
我们上面所有的例子中 谓词 总是 固定的, 然而我们在现实中处理变量时决定了谓词应该是 可变的. 下面我们来看看如何让谓词变化起来:
首先: 如果我们想在谓词表达式中使用变量,那么我们需要了解下列两种占位符:
%K: 用于动态传入属性名
%@: 用于动态设置属性值
除此之外, 还可以在谓词表达式中使用动态改变的属性值, 就像环境变量一样:
NSPredicate *pred = [NSPredicate predicateWithFormat:@"SELF CONTAINS $VALUE"];
上面表达式中, $VALUE是一个可以动态变化的值, 它最后其实是在字典中的一个key值, 所以可以根据你的需要写入不同的值, 但是必须有, 然后随着程序改变$VALUE这个 谓词 表达式的比较条件就可以动态改变.
例一:
NSArray *array = @[[ZLPersonModel personWithName:@"Jack" age:20 sex:ZLPersonSexMale],
[ZLPersonModel personWithName:@"Rose" age:22 sex:ZLPersonSexFemale],
[ZLPersonModel personWithName:@"Jackson" age:30 sex:ZLPersonSexMale],
[ZLPersonModel personWithName:@"Johnson" age:35 sex:ZLPersonSexMale]];
//定义一个property来存放属性名, 定义一个value来存放值
NSString *property = @"name";
NSString *value = @"Jack";
//该谓词的作用是如果property属性含有值value时就取出放入新的数组内, 这里是name包含Jack
NSPredicate *pred = [NSPredicate predicateWithFormat:@"%K CONTAINS %@", property, value];
NSArray *newArray = [array filteredArrayUsingPredicate:pred];
NSLog(@"newArray = %@",newArray);
//创建谓词, 属性名改为age, 要求这个age包含$VALUE字符串
NSPredicate *predTemp = [NSPredicate predicateWithFormat:@"%K > $VALUE", @"age"];
//指定$VALUE的值为25
NSPredicate *pred1 = [predTemp predicateWithSubstitutionVariables:@{@"VALUE":@25}];
NSArray *newArray1 = [array filteredArrayUsingPredicate:pred1];
NSLog(@"newArray1 = %@",newArray1);
//修改 $VALUE的值为32
NSPredicate *pred2 = [predTemp predicateWithSubstitutionVariables:@{@"VALUE":@32}];
NSArray *newArray2 = [array filteredArrayUsingPredicate:pred2];
NSLog(@"newArray2 = %@",newArray2);
备注:
//用常数值代替变量
//参数: 替换变量的字典
//返回: 替换变量后的谓词表达式
- (instancetype)predicateWithSubstitutionVariables:(NSDictionary<NSString *, id> *)variables;
输出结果为:
newArray = (
"[name = Jack, age = 20, sex = ZLPersonSexMale]",
"[name = Jackson age = 30, sex = ZLPersonSexMale]"
)
newArray1 = (
"[name = Jackson, age = 30, sex = ZLPersonSexMale]",
"[name = Johnson, age = 35, sex = ZLPersonSexMale]"
)
newArray2 = (
"[name = Johnson, age = 35, sex = ZLPersonSexMale]"
)