【原/转】iOS中非常强大的过滤器:NSPredicate
在APPLE的官方Demo:UICatalog中实现UISearchBar模糊搜索功能是这么做的:
1 - (void)viewDidLoad { 2 [super viewDidLoad]; 3 4 self.allResults = @[@"Here‘s", @"to", @"the", @"crazy", @"ones.", @"The", @"misfits.", @"The", @"rebels.", @"The", @"troublemakers.", @"The", @"round", @"pegs", @"in", @"the", @"square", @"holes.", @"The", @"ones", @"who", @"see", @"things", @"differently.", @"They‘re", @"not", @"fond", @"of", @"rules.", @"And", @"they", @"have", @"no", @"respect", @"for", @"the", @"status", @"quo.", @"You", @"can", @"quote", @"them,", @"disagree", @"with", @"them,", @"glorify", @"or", @"vilify", @"them.", @"About", @"the", @"only", @"thing", @"you", @"can‘t", @"do", @"is", @"ignore", @"them.", @"Because", @"they", @"change", @"things.", @"They", @"push", @"the", @"human", @"race", @"forward.", @"And", @"while", @"some", @"may", @"see", @"them", @"as", @"the", @"crazy", @"ones,", @"we", @"see", @"genius.", @"Because", @"the", @"people", @"who", @"are", @"crazy", @"enough", @"to", @"think", @"they", @"can", @"change", @"the", @"world,", @"are", @"the", @"ones", @"who", @"do."]; 5 6 self.visibleResults = self.allResults; 7 }
1 - (void)setFilterString:(NSString *)filterString { 2 _filterString = filterString; 3 4 if (!filterString || filterString.length <= 0) { 5 self.visibleResults = self.allResults; 6 } 7 else { 8 NSPredicate *filterPredicate = [NSPredicate predicateWithFormat:@"self contains[c] %@", filterString]; 9 self.visibleResults = [self.allResults filteredArrayUsingPredicate:filterPredicate]; 10 } 11 12 [self.tableView reloadData]; 13 }
其中,self.allResults是列表的全部结果,self.visibleResults是输入搜索词后出现的模糊匹配结果。流程如下图所示:
从上述代码可以看到,APPLE获取到模糊搜索结果所用的代码仅仅两行。由此可见,NSPredicate的功能不可小觑。这也是本文的目的,全方位地介绍一下在cocoa框架下的搜索匹配利器:NSPredicate。Cocoa框架中的NSPredicate用于查询,原理和用法都类似于SQL中的where,作用相当于数据库的过滤取。
1、初始化
NSPredicate *ca = [NSPredicate predicateWithFormat:(NSString *), ...];
那传入的初始化NSString到底要满足怎样的格式呢?
(1)比较运算符>,<,==,>=,<=,!= 可用于数值及字符串 例:@"number > 100" (2)范围运算符:IN、BETWEEN 例:@"number BETWEEN {1,5}" @"address IN {‘shanghai‘,‘beijing‘}" (3)字符串本身:SELF 例:@“SELF == ‘APPLE’" (4)字符串相关:BEGINSWITH、ENDSWITH、CONTAINS 例:@"name CONTAIN[cd] ‘ang‘" //包含某个字符串 @"name BEGINSWITH[c] ‘sh‘" //以某个字符串开头 @"name ENDSWITH[d] ‘ang‘" //以某个字符串结束 注:[c]不区分大小写,[d]不区分发音符号即没有重音符号,[cd]既不区分大小写,也不区分发音符号。 (5)通配符:LIKE 例:@"name LIKE[cd] ‘*er*‘" //*代表通配符,Like也接受[cd]. @"name LIKE[cd] ‘???er*‘" (6)正则表达式:MATCHES 例:NSString *regex = @"^A.+e$"; //以A开头,e结尾 @"name MATCHES %@",regex
2、使用
2.1 场景1:NSArray过滤,也就是文章开头的场景
NSArray *array = [[NSArray alloc]initWithObjects:@"beijing",@"shanghai",@"guangzou",@"wuhan", nil];
NSString *string = @"ang";
NSPredicate *pred = [NSPredicate predicateWithFormat:@"SELF CONTAINS %@",string];
NSLog(@"%@",[array filteredArrayUsingPredicate:pred]);
2.2 场景2:判断字符串首字母是否为字母
NSString *regex = @"[A-Za-z]+"; NSPredicate *predicate = [NSPredicate predicateWithFormat:@"SELF MATCHES %@", regex]; if ([predicate evaluateWithObject:aString]) { }
2.3 场景3:字符串替换
NSError* error = NULL; NSRegularExpression* regex = [NSRegularExpression regularExpressionWithPattern:@"(encoding=\")[^\"]+(\")" options:0 error:&error]; NSString* sample = @"<xml encoding=\"abc\"></xml><xml encoding=\"def\"></xml><xml encoding=\"ttt\"></xml>"; NSLog(@"Start:%@",sample); NSString* result = [regex stringByReplacingMatchesInString:sample options:0 range:NSMakeRange(0, sample.length) withTemplate:@"$1utf-8$2"]; NSLog(@"Result:%@", result);
2.4 场景4:截取字符串
//组装一个字符串,需要把里面的网址解析出来 NSString *urlString=@"<meta/><link/><title>1Q84 BOOK1</title></head><body>"; //NSRegularExpression类里面调用表达的方法需要传递一个NSError的参数。下面定义一个 NSError *error; //http+:[^\\s]* 这个表达式是检测一个网址的。(?<=title\>).*(?=</title)截取html文章中的<title></title>中内文字的正则表达式 NSRegularExpression *regex = [NSRegularExpression regularExpressionWithPattern:@"(?<=title\\>).*(?=</title)" options:0 error:&error]; if (regex != nil) { NSTextCheckingResult *firstMatch=[regex firstMatchInString:urlString options:0 range:NSMakeRange(0, [urlString length])]; if (firstMatch) { NSRange resultRange = [firstMatch rangeAtIndex:0]; //从urlString当中截取数据 NSString *result=[urlString substringWithRange:resultRange]; //输出结果 NSLog(@"->%@<-",result); } }
2.5 场景5:判断是否是手机号码或者电话号码
//组装一个字符串,需要把里面的网址解析出来 NSString *urlString=@"<meta/><link/><title>1Q84 BOOK1</title></head><body>"; //NSRegularExpression类里面调用表达的方法需要传递一个NSError的参数。下面定义一个 NSError *error; //http+:[^\\s]* 这个表达式是检测一个网址的。(?<=title\>).*(?=</title)截取html文章中的<title></title>中内文字的正则表达式 NSRegularExpression *regex = [NSRegularExpression regularExpressionWithPattern:@"(?<=title\\>).*(?=</title)" options:0 error:&error]; if (regex != nil) { NSTextCheckingResult *firstMatch=[regex firstMatchInString:urlString options:0 range:NSMakeRange(0, [urlString length])]; if (firstMatch) { NSRange resultRange = [firstMatch rangeAtIndex:0]; //从urlString当中截取数据 NSString *result=[urlString substringWithRange:resultRange]; //输出结果 NSLog(@"->%@<-",result); } }
2.6 场景6:验证邮箱、电话号码有效性
//是否是有效的正则表达式 +(BOOL)isValidateRegularExpression:(NSString *)strDestination byExpression:(NSString *)strExpression { NSPredicate *predicate = [NSPredicatepredicateWithFormat:@"SELF MATCHES %@", strExpression]; return [predicate evaluateWithObject:strDestination]; } //验证email +(BOOL)isValidateEmail:(NSString *)email { NSString *strRegex = @"[A-Z0-9a-z._%+-]+@[A-Za-z0-9.-]+\\.[A-Za-z]{1,5}"; BOOL rt = [CommonTools isValidateRegularExpression:email byExpression:strRegex]; return rt; } //验证电话号码 +(BOOL)isValidateTelNumber:(NSString *)number { NSString *strRegex = @"[0-9]{1,20}"; BOOL rt = [CommonTools isValidateRegularExpression:number byExpression:strRegex]; return rt; }
2.7 场景7:NSDate筛选
//日期在十天之内: NSDate *endDate = [[NSDate date] retain]; NSTimeInterval timeInterval= [endDate timeIntervalSinceReferenceDate]; timeInterval -=3600*24*10; NSDate *beginDate = [[NSDate dateWithTimeIntervalSinceReferenceDate:timeInterval] retain]; //对coredata进行筛选(假设有fetchRequest) NSPredicate *predicate_date = [NSPredicate predicateWithFormat:@"date >= %@ AND date <= %@", beginDate,endDate]; [fetchRequest setPredicate:predicate_date]; //释放retained的对象 [endDate release]; [beginDate release];
郑重声明:本站内容如果来自互联网及其他传播媒体,其版权均属原媒体及文章作者所有。转载目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。