前言
首先,我們需要知道何謂謂詞,讓我們看看官方的解釋:
The NSPredicate class is used to define logical conditions used to constrain a search either for a fetch or for in-memory filtering.
NSPredicate類是用來定義邏輯條件約束的獲取或內存中的過濾搜索。
可以使用謂詞來表示邏輯條件,用于描述對象持久性存儲在內存中的對象過濾。其實意思就是:我是一個過濾器,不符合條件的都滾開。
一、NSPredicate的基本語法
我們使用一門語言,無論是外語還是計算機語言,總是從語法開始的,這樣我們才能正確的把握邏輯。所以我們從語法開始說起。在這部分我們僅關心其語法的使用
只要我們使用謂詞(NSPredicate)都需要為謂詞定義謂詞表達式,而這個表達式必須是一個返回BOOL的值。
謂詞表達式由表達式、運算符和值構成。
1.比較運算符
比較運算符如下
=、==:判斷兩個表達式是否相等,在謂詞中=和==是相同的意思都是判斷,而沒有賦值這一說
NSNumber *testNumber = @123; NSPredicate *predicate = [NSPredicate predicateWithFormat:@"SELF = 123"]; if ([predicate evaluateWithObject:testNumber]) { NSLog(@"testString:%@", testNumber); }
我們可以看到輸出的內容為:
2016-01-07 11:12:27.281 PredicteDemo[4130:80412] testString:123
NSNumber *testNumber = @123;NSPredicate *predicate = [NSPredicate predicateWithFormat:@"SELF BETWEEN {100, 200}"]; if ([predicate evaluateWithObject:testNumber]) { NSLog(@"testString:%@", testNumber); } else { NSLog(@"不符合條件"); }
輸出結果為:
2016-01-07 11:20:39.921 PredicteDemo[4366:85408] testString:123
2.邏輯運算符
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);
輸出結果為:
2016-01-07 11:27:01.885 PredicteDemo[4531:89537] filterArray:(
3,
4
)
3.字符串比較運算符
注:字符串比較都是區分大小寫和重音符號的。如:café和cafe是不一樣的,Cafe和cafe也是不一樣的。如果希望字符串比較運算不區分大小寫和重音符號,請在這些運算符后使用[c],[d]選項。其中[c]是不區分大小寫,[d]是不區分重音符號,其寫在字符串比較運算符之后,比如:name LIKE[cd] 'cafe',那么不論name是cafe、Cafe還是café上面的表達式都會返回YES。
4.集合運算符
NSArray *filterArray = @[@"ab", @"abc"]; NSArray *array = @[@"a", @"ab", @"abc", @"abcd"]; NSPredicate *predicate = [NSPredicate predicateWithFormat:@"NOT (SELF IN %@)", filterArray]; NSLog(@"%@", [array filteredArrayUsingPredicate:predicate]);
代碼的作用是將array中和filterArray中相同的元素去除,輸出為:
2016-01-07 13:17:43.669 PredicteDemo[6701:136206] (
a,
abcd
)
5.直接量
在謂詞表達式中可以使用如下直接量
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:<#(nonnull NSString *), ...#>];
下面我們通過幾個簡單的例子來看看它該如何使用:
首先我們需要定義一個模型,因為示例中需要用到它
ZLPersonModel.h
#import <Foundation/Foundation.h>typedef NS_ENUM(NSInteger, ZLPersonSex) { ZLPersonSexMale = 0, ZLPersonSexFamale};@interface ZLPersonModel : NSObject/** NSString 姓名 */@property (nonatomic, copy) NSString *name;/** NSUInteger 年齡 */@property (nonatomic, assign, readonly) NSUInteger age;/** ZLPersonSex 性別 */@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{ if (self = [super init]) { _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];}- (NSString *)description{ return [NSString stringWithFormat:@"[name = %@, age = %ld, sex = %ld]", self.name, self.age, self.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]);
看到這里我們會發現evaluateWithObject:方法返回的是一個BOOL值,如果符合條件就返回YES,不符合就返回NO。而即使是最簡單的使用也有一些大用處,比如以前我們寫判斷手機號碼、郵編等等,像我就喜歡用John Engelhart大神的RegexKitLite,然而由于年代久遠需要導入libicucore.dylib庫(xcode7為libicucore.tbd)且由于是mrc又需要添加-fno-objc-arc,至此我們才能使用。然而使用謂詞讓我們可以用同樣簡潔的代碼實現相同的功能
例二:
(判斷手機號是否正確)
- (BOOL)checkPhoneNumber:(NSString *)phoneNumber{ NSString *regex = @"^[1][3-8]//d{9}$"; NSPredicate *pred = [NSPredicate predicateWithFormat:@"SELF MATCHES %@", regex]; return [pred evaluateWithObject:phoneNumber];}
例三:檢測字符串中是否有特殊字符
- (BOOL)checkSpecialCharacter:(NSString *)string{ NSString *regex = @"[`~!@#$^&*()=|{}':;',//[//].<>/?~!@#¥……&*()――|{}【】‘;:”“'。,、?]+"; NSPredicate *pred = [NSPredicate predicateWithFormat:@"SELF MATCHES %@", regex]; return [pred evaluateWithObject:string];}
2.使用謂詞過濾集合
此部分是我們需要掌握的重點,因為從這里我們就可以看到謂詞的真正的強大之處
其實謂詞本身就代表了一個邏輯條件,計算謂詞之后返回的結果永遠為BOOL類型的值。而謂詞最常用的功能就是對集合進行過濾。當程序使用謂詞對集合元素進行過濾時,程序會自動遍歷其元素,并根據集合元素來計算謂詞的值,當這個集合中的元素計算謂詞并返回YES時,這個元素才會被保留下來。請注意程序會自動遍歷其元素,它會將自動遍歷過之后返回為YES的值重新組合成一個集合返回。
其實類似于我們使用tableView設置索引時使用的下段代碼
- (NSArray<NSString *> *)sectionIndexTitlesForTableView:(UITableView *)tableView{ return [self.cityGroup valueForKey:@"title"];}
中的[self.cityGroup valueForKey:@"title"]。它的作用是遍歷所有title并將得到的值組成新的數組。
- (NSArray<ObjectType> *)filteredArrayUsingPredicate:(NSPredicate *)predicate
:使用指定的謂詞過濾NSArray集合,返回符合條件的元素組成的新集合- (void)filterUsingPredicate:(NSPredicate *)predicate
:使用指定的謂詞過濾NSMutableArray,剔除集合中不符合條件的元素- (NSSet<ObjectType> *)filteredSetUsingPredicate:(NSPredicate *)predicate NS_AVAILABLE(10_5, 3_0)
:作用同NSArray中的方法- (void)filterUsingPredicate:(NSPredicate *)predicate NS_AVAILABLE(10_5, 3_0)
:作用同NSMutableArray中的方法。下面讓我們來看幾個例子:
例一:
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:ZLPersonSexFamale], [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);
輸出為
2016-01-07 16:50:09.510 PredicteDemo[13660:293822] arrayM:(
60,
70
)
2016-01-07 16:50:09.511 PredicteDemo[13660:293822] (
"[name = Jackson, age = 30, sex = 0]",
"[name = Johnson, age = 35, sex = 0]"
)
從這個例子我們就可以看到NSPredicate有多么強大,如果讓我們用其他的方法來實現又是一大堆if...else。
讓我們來回顧一下上面的從第二個數組中去除第一個數組中相同的元素
例二:
NSArray *filterArray = @[@"ab", @"abc"]; NSArray *array = @[@"a", @"ab", @"abc", @"abcd"]; NSPredicate *predicate = [NSPredicate predicateWithFormat:@"NOT (SELF IN %@)", filterArray]; NSLog(@"%@", [array filteredArrayUsingPredicate:predicate]);
輸出為:
2016-01-07 13:17:43.669 PredicteDemo[6701:136206] (
a,
abcd
)
如果我們不用NSPredicate的話,肯定又是各種if...else,for循環等等??梢钥闯鯪SPredicate的出現為我們節省了大量的時間和精力。
3.在謂詞中使用占位符參數
我們上面所有的例子中謂詞總是固定的,然而我們在現實中處理變量時決定了謂詞應該是可變的。下面我們來看看如果讓謂詞變化起來。
首先如果我們想在謂詞表達式中使用變量,那么我們需要了解下列兩種占位符
其實相當于變量名與變量值
除此之外,還可以在謂詞表達式中使用動態改變的屬性值,就像環境變量一樣
NSPredicate *pred = [NSPredicate predicateWithFormat:@"SELF CONTAINS $VALUE"];
上述表達式中,開頭,隨著程序改變$VALUE這個謂詞表達式的比較條件就可以動態改變。
下面我們通過一個例子來看看這三個重要的占位符應該如何使用
例一:
NSArray *array = @[[ZLPersonModel personWithName:@"Jack" age:20 sex:ZLPersonSexMale], [ZLPersonModel personWithName:@"Rose" age:22 sex:ZLPersonSexFamale], [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);
輸出為
2016-01-07 17:28:02.062 PredicteDemo[14542:309494] newArray:(
"[name = Jack, age = 20, sex = 0]",
"[name = Jackson, age = 30, sex = 0]"
)
2016-01-07 17:28:02.063 PredicteDemo[14542:309494] newArray1:(
"[name = Jackson, age = 30, sex = 0]",
"[name = Johnson, age = 35, sex = 0]"
)
2016-01-07 17:28:02.063 PredicteDemo[14542:309494] newArray2:(
"[name = Johnson, age = 35, sex = 0]"
)
從上例中我們主要可以看出來%K和$VALUE的含義。
那么至此NSPredicate就到到此介紹完畢。
因為本人水平有限,如果有什么好的建議請聯系我。
總結
以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對武林網的支持。
新聞熱點
疑難解答