NSPredicate

来源:互联网 发布:北师大网络教育电话 编辑:程序博客网 时间:2024/04/20 05:09

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


定义(最常用到的方法):

[plain] view plaincopy
  1. NSPredicate *ca = [NSPredicate predicateWithFormat:(NSString *), ...];  

例子1:

(1)对NSArray进行过滤 

[plain] view plaincopy
  1. NSArray *array = [[NSArray alloc]initWithObjects:@"beijing",@"shanghai",@"guangzou",@"wuhan", nil];    
  2. NSString *string = @"ang";    
  3. NSPredicate *pred = [NSPredicate predicateWithFormat:@"SELF CONTAINS %@",string];    
  4. NSLog(@"%@",[array filteredArrayUsingPredicate:pred]);    

例子2:

[plain] view plaincopy
  1. #import <Foundation/Foundation.h>  
  2. @interface Person: NSObject{  
  3.     int pid;  
  4.     NSString *name;  
  5.     float height;  
  6. }  
  7. -(void) setPid: (int) pid;  
  8. -(void) setName: (NSString*) name;  
  9. -(void) setHeight: (float) height;  
  10. -(int) pid;  
  11. -(NSString*) name;  
  12. -(float) height;  
  13. @end  

[plain] view plaincopy
  1. #import "Person.h"  
  2.   
  3. @implementation Person  
  4. -(void) setPid: (int) p{  
  5.     pid=p;  
  6. }  
  7. -(void) setName: (NSString*) n{  
  8.     [n retain];  
  9.     [name release];  
  10.     name=n;  
  11. }  
  12. -(void) setHeight: (float) h{  
  13.     height=h;  
  14. }  
  15. -(int) pid{  
  16.     return pid;  
  17. }  
  18. -(NSString*) name{  
  19.     return name;  
  20. }  
  21. -(float) height{  
  22.     return height;  
  23. }  
  24. -(void) dealloc{  
  25.     [name release];  
  26.     [super dealloc];  
  27. }  
  28. @end  

[plain] view plaincopy
  1. int main(int argc, char *argv[])  
  2. {  
  3.     NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];  
  4.     //实例化三个Person,并放入数组。  
  5.     NSMutableArray *array=[NSMutableArray arrayWithCapacity: 5];  
  6.     Person *person1=[[Person alloc] init];  
  7.     [person1 setPid: 1];  
  8.     [person1 setName: @"Name1"];  
  9.     [person1 setHeight: 168];  
  10.     [array addObject: person1];  
  11.     Person *person2=[[Person alloc] init];  
  12.     [person2 setPid: 2];  
  13.     [person2 setName: @"Name2"];  
  14.     [person2 setHeight: 178];  
  15.     [array addObject: person2];  
  16.     Person *person3=[[Person alloc] init];  
  17.     [person3 setPid: 3];  
  18.     [person3 setName: @"Name3"];  
  19.     [person3 setHeight: 188];  
  20.     [array addObject: person3];  
  21.     //创建谓词,条件是pid>1 并且height<188.0。其实谓词也是基于KVC 的,也就是如果pid 在person 的成员变量xxx 中,那么此处要写成xxx.pid>1。  
  22.     NSPredicate *pre = [NSPredicate predicateWithFormat:  
  23.                         @" pid>1 and height<188.0"];  
  24.     int i=0;  
  25.     for(;i<[array count];i++){  
  26.         Person *person=[array objectAtIndex: i];  
  27.         //遍历数组,输出符合谓词条件的Person 的name。  
  28.         if ([pre evaluateWithObject: person]) {  
  29.             NSLog(@"%@",[person name]);  
  30.         }  
  31.     }  
  32.     [person1 release];  
  33.     [person2 release];  
  34.     [person3 release];  
  35.     [pool release];  
  36.     return 0;  
  37. }  

我们看到创建谓词使用类方法predicateWithFormat: (NSString*) format,

format 里的东西真的和SQL 的where 条件差不多。

另外,参数format 与NSLog 的格式化模版差不多,如果1 和188.0 是传递过来的参数,你可以写成如下的形式:

@"pid>%d and height<%f",1,188.0
(1)比较运算符>,<,==
可用于数值及字符串
例:@"number > 100"
(1.) 逻辑运算符:AND、OR、NOT
这几个运算符计算并、或、非的结果。
(2.) 范围运算符:BETWEEN、IN
例:
@”pid BETWEEN {1,5}”
@"name IN {'Name1','Name2'}"
(3.) 占位符:
NSPredicate *preTemplate = [NSPredicate predicateWithFormat:@"name==$NAME"];
NSDictionary *dic=[NSDictionary dictionaryWithObjectsAndKeys:
@"Name1", @"NAME",nil];
NSPredicate *pre=[preTemplate predicateWithSubstitutionVariables: dic];
占位符就是字段对象里的key,因此你可以有多个占位符,只要key 不一样就可以了。
(4.) 快速筛选数组:
前面我们都是使用谓词逐个判断数组内的对象是否符合,其实数组本身有更为便捷的方法,
直接筛选出一个符合谓词的新数组。
NSPredicate *pre = [NSPredicate predicateWithFormat:@"pid>1"];
NSMutableArray *arrayPre=[array filteredArrayUsingPredicate: pre];
NSLog(@"%@",[[arrayPre objectAtIndex: 0] name]);
(5.) 字符串运算符:
BEGINSWITH、ENDSWITH、CONTAINS 分别表示是否以某字符串开头、结尾、包含。
他们可以与c、d 连用,表示是否忽略大小写、是否忽略重音字母(字母上方有声调标号)。
例:
@”name BEGINSWITH[cd] ‘He’”
判断name 是否以He 开头,并且忽略大小写、忽略重音字母。
(6.) LIKE 运算符:
LIKE 使用?表示一个字符,*表示多个字符,也可以与c、d 连用。
例:
@”name LIKE ‘???er*’” 与Paper Plane 相匹配。
(7.) SELF:
前面的数组中放的都是对象,如果数组放的都是字符串(或者是其他没有属性的类型),该
怎么写谓词呢?这里我们使用SELF。
例:
NSArray *arrays=[NSArray arrayWithObjects: @"Apple", @"Google", @"MircoSoft", nil];
NSPredicate *pre2 = [NSPredicate predicateWithFormat:@"SELF=='Apple'"];
(8.) 正则表达式:
NSPredicate 使用MATCHES 匹配正则表达式,正则表达式的写法采用international components
for Unicode (ICU)的正则语法。
例:
NSString *regex = @"^A.+e$";//以A 开头,以e 结尾的字符。
NSPredicate *pre= [NSPredicate predicateWithFormat:@"SELF MATCHES %@", regex];
if([pre evaluateWithObject: @"Apple"]){
printf("YES\n");
}else{
printf("NO\n");
}


例子3:

一般来说这种情况还是蛮多的,比如你从文件中读入了一个array1,然后想把程序中的一个array2中符合array1中内容的元素过滤出来。

正 常傻瓜一点就是两个for循环,一个一个进行比较,这样效率不高,而且代码也不好看。

其实一个循环或者无需循环就可以搞定了,那就需要用搞 NSPredicate这个类了~膜拜此类~

1)例子一,一个循环

NSArray *arrayFilter = [NSArray arrayWithObjects:@"pict", @"blackrain", @"ip", nil];

NSArray *arrayContents = [NSArray arrayWithObjects:@"I am a picture.", @"I am a guy", @"I am gagaga", @"ipad", @"iphone", nil];

我想过滤arrayContents的话只要循环 arrayFilter就好了

int i = 0, count = [arrayFilter count];

for(i = 0; i < count; i ++)

{

NSString *arrayItem = (NSString *)[arrayFilter objectAtIndex:i];

NSPredicate *filterPredicate = [[NSPredicate predicateWithFormat:@"SELF CONTAINS %@", arrayItem];

NSLog(@"Filtered array with filter %@, %@", arrayItem, [arrayContents filteredArrayUsingPredicate:filterPredicate]);

}

当然以上代码中arrayContent最好用mutable 的,这样就可以直接filter了,NSArray是不可修改的。

2)例子二,无需循环

NSArray *arrayFilter = [NSArray arrayWithObjects:@"abc1", @"abc2", nil];

NSArray *arrayContent = [NSArray arrayWithObjects:@"a1", @"abc1", @"abc4", @"abc2", nil];

NSPredicate *thePredicate = [NSPredicate predicateWithFormat:@"NOT (SELF in %@)", arrayFilter];

[arrayContent filterUsingPredicate:thePredicate];


这样arrayContent过滤出来的就是不包含 arrayFilter中的所有item了。


3)生成文件路径下文件集合列表

NSFileManager *fileManager = [NSFileManager defaultManager];
    NSString *defaultPath = [[NSBundle mainBundle] resourcePath];
    NSError *error;
    NSArray *directoryContents = [fileManager contentsOfDirectoryAtPath:defaultPath error:&error]


4)match的用法

    1. 简单比较

    NSString *match = @"imagexyz-999.png";
    NSPredicate *predicate = [NSPredicate predicateWithFormat:@"SELF == %@", match];
    NSArray *results = [directoryContents filteredArrayUsingPredicate:predicate];
2. match里like的用法(类似Sql中的用法)
  NSString *match = @"imagexyz*.png";
    NSPredicate *predicate = [NSPredicate predicateWithFormat:@"SELF like %@", match];
    NSArray *results = [directoryContents filteredArrayUsingPredicate:predicate];
3. 大小写比较
  [c]表示忽略大小写,[d]表示忽略重音,可以在一起使用,如下:
NSString *match = @"imagexyz*.png";
    NSPredicate *predicate = [NSPredicate predicateWithFormat:@"SELF like[cd] %@", match];
    NSArray *results = [directoryContents filteredArrayUsingPredicate:predicate];
4.使用正则 
NSString *match = @"imagexyz-\\d{3}\\.png";  //imagexyz-123.png
    NSPredicate *predicate = [NSPredicate predicateWithFormat:@"SELF matches %@", match];
    NSArray *results = [directoryContents filteredArrayUsingPredicate:predicate];


总结:

1) 当使用聚合类的操作符时是可以不需要循环的

2)当使用单个比较类的操作符时可以一个循环来搞定

PS,例子 一中尝试使用[@"SELF CONTAINS %@", arrayFilter] 来过滤会挂调,因为CONTAINS时字符串比较操作符,不是集合操作符。







NSPredicate源代码:

[plain] view plaincopy
  1. #import <Foundation/NSObject.h>  
  2. #import <Foundation/NSArray.h>  
  3. #import <Foundation/NSSet.h>  
  4.   
  5. NS_CLASS_AVAILABLE(10_4, 3_0)  
  6. @interface NSPredicate : NSObject <NSCoding, NSCopying> {  
  7.     void *_reserved;  
  8. }  
  9. + (NSPredicate *)predicateWithFormat:(NSString *)predicateFormat argumentArray:(NSArray *)arguments;  
  10. + (NSPredicate *)predicateWithFormat:(NSString *)predicateFormat, ...;  
  11. + (NSPredicate *)predicateWithFormat:(NSString *)predicateFormat arguments:(va_list)argList;  
  12.   
  13. + (NSPredicate *)predicateWithValue:(BOOL)value;     
  14.   
  15. #if NS_BLOCKS_AVAILABLE  
  16. + (NSPredicate*)predicateWithBlock:(BOOL (^)(id evaluatedObject, NSDictionary *bindings))block NS_AVAILABLE(10_6, 4_0);   
  17. #endif  
  18.   
  19. - (NSString *)predicateFormat;      
  20.   
  21. - (NSPredicate *)predicateWithSubstitutionVariables:(NSDictionary *)variables;      
  22.   
  23. - (BOOL)evaluateWithObject:(id)object;      
  24.   
  25. - (BOOL)evaluateWithObject:(id)object substitutionVariables:(NSDictionary *)bindings NS_AVAILABLE(10_5, 3_0);   
  26.   
  27. @end  
  28.   
  29. @interface NSArray (NSPredicateSupport)  
  30. - (NSArray *)filteredArrayUsingPredicate:(NSPredicate *)predicate;      
  31. @end  
  32.   
  33. @interface NSMutableArray (NSPredicateSupport)  
  34. - (void)filterUsingPredicate:(NSPredicate *)predicate;      
  35. @end  
  36.   
  37.   
  38. @interface NSSet (NSPredicateSupport)  
  39. - (NSSet *)filteredSetUsingPredicate:(NSPredicate *)predicate NS_AVAILABLE(10_5, 3_0);      
  40. @end  
  41.   
  42. @interface NSMutableSet (NSPredicateSupport)  
  43. - (void)filterUsingPredicate:(NSPredicate *)predicate NS_AVAILABLE(10_5, 3_0);      
  44. @end  


原文来自:http://blog.csdn.net/zhulei1018/article/details/6777220

原文来自:http://www.cnblogs.com/MarsGG/articles/1949239.html

原创粉丝点击