( 转 )NSPredicate-谓词使用总结

原文链接地址:http://blog.csdn.net/st646889325/article/details/53501286
简介

NSPredicate类主要用来指定过滤器的条件,该对象可以准确的描述所需条件,对每个对象通过谓词进行筛选,判断是否与条件相匹配。谓词是指在计算机中表示计算真假值的函数。原理和用法都类似于SQL查询中的where,作用相当于数据库的过滤取。主要用于从集合中分拣出符合条件的对象或者数据模型,也可以用于字符串的正则匹配.

一般的, NSPredicate的筛选过滤的条件可以是, 逻辑运算符号(> , < , =),范围运算符(IN,BETWEEN),字符的包含/匹配/模糊搜索(BEGINSWITH,ENDSWITH,CONTAINS,LIKE), 正则表达式(MATCHES)等.

NSPredicate用法

1.谓词过滤筛选的过程

1.创建NSPredicate

NSPredicate *predicate = [NSPredicate predicateWithFormat:@"过滤条件"];

2.判断指定的对象是否满足NSPredicate创建的过滤条件

 [predicate evaluateWithObject:model];

3.将array通过创建的predicate进行过滤,并返回符合条件的数据

 NSArray *resultArr = [arrayfilteredArrayUsingPredicate:predicate];

这些方法既适用于字符串这样的简单对象的过滤, 又适用于数据模型的过滤.

举例:

对简单的字符串数组进行包含过滤

NSArray*array=@[@"abc",@"adbzc",@"adboc",@"skenabc",@"aksalkjjbc2c",@"111abc",@"a22bc4",@"abc4444",@"asdfad",@"alkmbc",@"abcopjoj",@"abjoc",@"khujabc",@"abc9074"];  
NSPredicate *predicate = [NSPredicate predicateWithFormat:@"SELF CONTAINS[cd] %@", @"abc"];
NSArray*resultArr = [array filteredArrayUsingPredicate:predicate];
NSLog(@"%@",resultArr);

谓词过滤的筛选条件

1.逻辑运算符号(> , < , = , >= , <=)

还可以跟逻辑运算符一起使用的: &&, || ,AND, OR 谓词不区分大小写

NSPredicate*predicate= [NSPredicate predicateWithFormat:@"age >20"]; 
NSPredicate*predicate= [NSPredicate predicateWithFormat:@"height > 180 && age > 10"];  
NSPredicate*predicate= [NSPredicate predicateWithFormat:@"height >180OR age >10"];

2.范围运算符(IN,BETWEEN)

NSPredicate*predicate= [NSPredicate predicateWithFormat:"age BETWEEN {1,5}"];  
NSPredicate*predicate= [NSPredicate predicateWithFormat:@"name IN {'abc','def','123'}"]; 

3.字符串之开头和结尾(BEGINSWITH,ENDSWITH)

BEGINSWITH:以**开头

ENDSWITH:以**结尾

//name以N打头

NSPredicate *predicate =[NSPredicate predicateWithFormat:@"name BEGINSWITH 'N'"];  
NSPredicate *predicate =[NSPredicate predicateWithFormat:@"name BEGINSWITH[c] 'M'"];
NSPredicate *predicate =[NSPredicate predicateWithFormat:@"name BEGINSWITH[d] 'C'"];  
NSPredicate *predicate =[NSPredicate predicateWithFormat:@"name BEGINSWITH[cd] 'w'"];

注:字符串类的筛选条件中, [c]不区分大小写[d]不区分发音符号即没有重音符号[cd]既不区分大小写,也不区分发音符号。

4.字符串之包含和模糊查询(CONTAINS,LIKE)

CONTAINS:包含某个字符串

LIKE:模糊查询

NSPredicate*predicate= [NSPredicate predicateWithFormat:@"name CONTAINS'N'"]; 
NSPredicate*predicate= [NSPredicate predicateWithFormat:@"name LIKE '*N*'"];
NSPredicate*predicate= [NSPredicate predicateWithFormat:@"name LIKE[cd]'???er*"];

注:LIKE中的?表示一个任意字符, *表示通配符

5.字符串匹配查询(SELF)

以上说的都是对象中的属性匹配的筛选条件,如果数组中都是字符串,即非属性匹配查询, 需要用到SELF.

NSPredicate*predicate= [NSPredicate predicateWithFormat:@"SELF CONTAINS[cd]%@", @"abc"];

6.正则表达式(MATCHES)

NSPredicate 使用MATCHES 匹配正则表达式,正则表达式的写法采用international components

for Unicode (ICU)的正则语法。

NSString*regex=@"^A.+e$";//以A 开头,以e 结尾的字符。

NSPredicate*pre= [NSPredicate predicateWithFormat:@"SELF MATCHES%@", regex];

举例

例一

NSArray *array=[NSArray arrayWithObjects:person1,person2,person3,person4,...,nil];  
//方法一:手动过滤出age小于20的person数组
NSPredicate *predicate = [NSPredicate predicateWithFormat:@"age < 20"];  
for(Person *person in array){  
    if([predicate evaluateWithObject:person]){ 
    //判断指定的对象是否满足
    }  
}  
//方法二:直接获取过滤出age小于20的person数组
NSArray *persons = [array filteredArrayUsingPredicate:predicate];//获取所有age小于20的person 

例二

将self.dataSourceArray模型数组中, 对 groupname 和 diseasename两个分别进行筛选, 将筛选模型添加到self.resulArr中, 并去重.

NSPredicate *namePredicate = [NSPredicate predicateWithFormat:@"groupname CONTAINS[cd] %@",searchString];
NSArray *nameArr = [self.dataSourceArray filteredArrayUsingPredicate:namePredicate];
NSPredicate *diseasePredicate = [NSPredicate predicateWithFormat:@"diseasemc CONTAINS[cd] %@",searchString];
NSArray *diseaseArr = [self.dataSourceArray filteredArrayUsingPredicate:diseasePredicate];
NSMutableSet *set = [NSMutableSet setWithArray:nameArr];
[set addObjectsFromArray:diseaseArr];
self.resultArr = [set sortedArrayUsingDescriptors:@[[[NSSortDescriptor alloc] initWithKey:@"groupname" ascending:YES]]];

总结和注意

1.NSPredicate既适用于属性匹配,对数据模型或者dic进行过滤; 又适用于简单字符串的非属性匹配. 区别只是谓词的条件用法不同.
2.在字符串筛选的谓词条件中, [c]不区分大小写[d]不区分发音符号即没有重音符号[cd]既不区分大小写,也不区分发音符号。
3.在例子中, filteredArrayUsingPredicate语句是进行过滤的执行. 然后, 我们进入到NSPredicate类中, 可以发现, NSPredicateSupportNSArray,NSMutableArray,NSSet,NSMutableSet,NSOrderedSet,NSMutableOrderedSet这些类都是适用的.

@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

@interface NSOrderedSet<ObjectType> (NSPredicateSupport)
- (NSOrderedSet<ObjectType> *)filteredOrderedSetUsingPredicate:(NSPredicate *)p NS_AVAILABLE(10_7, 5_0);    // evaluate a predicate against an ordered set of objects and return a filtered ordered set
@end

@interface NSMutableOrderedSet<ObjectType> (NSPredicateSupport)
- (void)filterUsingPredicate:(NSPredicate *)p NS_AVAILABLE(10_7, 5_0);  // evaluate a predicate against an ordered set of objects and filter the mutable ordered set directly
@end
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,362评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,330评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,247评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,560评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,580评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,569评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,929评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,587评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,840评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,596评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,678评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,366评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,945评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,929评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,165评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,271评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,403评论 2 342

推荐阅读更多精彩内容