NSPredicate iOS中非常强大的过滤器

在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、初始化

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

那传入的初始化NSString到底要满足怎样的格式呢?

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
(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过滤,也就是文章开头的场景

1
2
3
4
NSArray *array = [[ NSArray alloc]initWithObjects:@ "beijing" ,@ "shanghai" ,@ "guangzou" ,@ "wuhan" , nil ]; 
NSString *string = @ "ang"
NSPredicate *pred = [ NSPredicate predicateWithFormat:<span style= "color: #ff0000;" >@ "SELF CONTAINS %@" ,string</span>]; 
NSLog (@ "%@" ,[array filteredArrayUsingPredicate:pred]); 

2.2 场景2:判断字符串首字母是否为字母

1
2
3
4
5
NSString *regex = @ "[A-Za-z]+"
NSPredicate *predicate = [ NSPredicate predicateWithFormat:@ "SELF MATCHES %@" , regex]; 
   
if ([predicate evaluateWithObject:aString]) { 

2.3 场景3:字符串替换

1
2
3
4
5
6
7
8
9
10
11
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:截取字符串

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
//组装一个字符串,需要把里面的网址解析出来 
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:判断是否是手机号码或者电话号码

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
//组装一个字符串,需要把里面的网址解析出来 
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:验证邮箱、电话号码有效性

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
//是否是有效的正则表达式
 
+( 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筛选

1
2
3
4
5
6
7
8
9
10
11
12
13
//日期在十天之内:
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];

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是一个使用NSPredicate的Swift代码示例,用于将大量数据按时间分组: ```swift struct DataItem { let name: String let timeStamp: TimeInterval // 时间戳 } func groupDataByTime(_ data: [DataItem]) -> [[DataItem]] { // 将数据按时间先后排序 let sortedData = data.sorted { $0.timeStamp < $1.timeStamp } // 创建空字典 var groups = [String: [DataItem]]() // 使用NSPredicate过滤并分组数据 let dateFormatter = DateFormatter() dateFormatter.dateFormat = "yyyy-MM-dd" for item in sortedData { let date = Date(timeIntervalSince1970: item.timeStamp) let dateString = dateFormatter.string(from: date) let predicate = NSPredicate(format: "SELF.dateString == %@", dateString) let filteredArray = (groups as NSDictionary).filtered(using: predicate) if var items = filteredArray.first?.value as? [DataItem] { items.append(item) groups[dateString] = items } else { groups[dateString] = [item] } } // 将分组后的数据按时间先后顺序排序 let sortedGroups = groups.sorted { $0.key < $1.key } // 将所有分组添加到一个数组 var result = [[DataItem]]() for (_, items) in sortedGroups { result.append(items) } return result } ``` 在这个示例,我们同样定义了一个`DataItem`结构体来表示数据的名称和时间戳。`groupDataByTime`函数首先将数据按时间先后排序,然后使用`NSPredicate`过滤并分组数据。我们指定`NSPredicate`的格式为`SELF.dateString == %@`,其`dateString`是我们根据时间戳计算得到的时间字符串,`%@`是用于替换的占位符。通过对字典进行过滤,我们可以得到一个包含指定时间字符串的数组,然后将新的数据项添加到这个数组。最后,我们将所有分组添加到一个大数组,按时间先后顺序排序,并返回结果。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值