iOS NSPredicate 謂詞

前言

NSPredicate 謂詞, 其實(shí) NSPredicate 有時候完全可以被 [for...][if..else] 替換來實(shí)現(xiàn)功能, 但會造成代碼不夠優(yōu)雅, 一堆的判斷循環(huán) 有時一句 NSPredicate 便可實(shí)現(xiàn). 今天作者就聊一聊謂詞的使用.

一 NSPredicate 理解說明

NSPredicate : 中文直譯謂詞, 用來定義邏輯條件約束的搜索 或 內(nèi)存中的過濾。

  • 如同語法中的謂詞, 如 [3 大于 2]中"大于"就是一個就是謂詞. 簡單點(diǎn)說 它是邏輯判斷, 如同過濾器, 篩選你所需要的.

  • NSPredicate[for...][if..else] 功能有相同的地方, 單純比較在手機(jī)上運(yùn)行效率, 作者查閱過[度娘] [谷哥], 只簡單說會提升效率, 但未找到具體數(shù)據(jù)說明, 作者便不引用了.

  • 按作者個人理解 NSPredicate 如同專車服務(wù),抵達(dá)目的地下車; [for...][if..else]便是公交服務(wù), 你自己判斷在哪一站下車. 所以NSPredicate有一定的便捷性, [for...][if..else]有更廣的適用性.

二 NSPredicate 語法說明

//一般初始化
 NSPredicate *pred = [NSPredicate predicateWithFormat:@"...", ...];
//與具體對象 進(jìn)行篩選判斷, 返回為BOOL值
[pred evaluateWithObject:...]

  • 舉例
//SELF支持小寫, 代表正在被判斷的對象自身
  NSNumber *num = @999;
  NSPredicate *pred = [NSPredicate predicateWithFormat:@"SELF = 999"];
  if ([pred evaluateWithObject:num]) {
      NSLog(@"%@", num);
  }
//結(jié)果打印
999

一般很少這么用, 因為這么寫都不如 [if..else] 便捷, 下面就講一下 NSPredicate 的用法.

- 0. 集合類型方法說明
NSArray提供 : - (NSArray * )filteredArrayUsingPredicate:(NSPredicate *)predicate
NSMutableArray提供 : - (void)filterUsingPredicate:(NSPredicate *)predicate
NSSet提供 : - (NSSet *)filteredSetUsingPredicate:(NSPredicate *)predicate
NSMutableSet提供 : - (void)filterUsingPredicate:(NSPredicate *)predicate
- 1. 比較運(yùn)算符 (以數(shù)組舉例說明)
//創(chuàng)建 Person 類數(shù)組
    Person *p0 = [Person personName:@"ZhangSan" withAge:20 withSex:@"man"];
    Person *p1 = [Person personName:@"HanMeiMei" withAge:12 withSex:@"woman"];
    Person *p2 = [Person personName:@"LiLei" withAge:13 withSex:@"man"];
    Person *p3 = [Person personName:@"XiaoHua" withAge:13 withSex:@"woman"];
    NSArray *arr = @[p0, p1, p2, p3];
 /** 比較運(yùn)算符 */
    NSPredicate *pred = [NSPredicate predicateWithFormat:@"age < %@", @20];
    NSArray *resultArr = [arr filteredArrayUsingPredicate:pred];
    NSLog(@"年齡小于20 :%@", resultArr);
    
    
    pred = [NSPredicate predicateWithFormat:@"sex = 'woman' && age = 13"];
    resultArr = [arr filteredArrayUsingPredicate:pred];
    NSLog(@"年齡為13的女性 : %@", resultArr);

   
    NSMutableArray *arrayM = [@[@20, @40, @50, @30, @60, @70] mutableCopy];
//  可以用 'BETWEEN {30, 50}' 代替 '>'
//  pred = [NSPredicate predicateWithFormat:@"SELF > 50"];
    pred = [NSPredicate predicateWithFormat:@"SELF BETWEEN {30, 50}"];
    [arrayM filterUsingPredicate:pred];
    NSLog(@"可變數(shù)組過濾 : %@", arrayM);
//結(jié)果打印

年齡小于20 :(
   "HanMeiMei, 12, woman",
    "LiLei, 13, man",
    "XiaoHua, 13, woman"
)


年齡為13的女性 :(
   "XiaoHua, 13, woman"
)

可變數(shù)組過濾 : (
    40,
    50,
    30
)
- 2. 字符串運(yùn)算符
  • BEGINSWITH:檢查某個字符串是否以指定的字符串開頭(如判斷字符串是否以a開頭:BEGINSWITH 'a')
  • ENDSWITH:檢查某個字符串是否以指定的字符串結(jié)尾
  • CONTAINS:檢查某個字符串是否包含指定的字符串
  • LIKE:檢查某個字符串是否匹配指定的字符串模板。其之后可以跟?代表一個字符和代表任意多個字符兩個通配符。比如"name LIKE 'ac'",這表示name的值中包含ac則返回YES;"name LIKE '?ac'",表示name的第2、3個字符為ac時返回YES。
  • MATCHES:檢查某個字符串是否匹配指定的正則表達(dá)式。雖然正則表達(dá)式的執(zhí)行效率是最低的,但其功能是最強(qiáng)大的,也是我們最常用的。
  • 注:字符串比較都是區(qū)分大小寫和重音符號的。如:café和cafe是不一樣的,Cafe和cafe也是不一樣的。如果希望字符串比較運(yùn)算不區(qū)分大小寫和重音符號,請在這些運(yùn)算符后使用[c],[d]選項。其中[c]是不區(qū)分大小寫,[d]是不區(qū)分重音符號,其寫在字符串比較運(yùn)算符之后,比如:name LIKE[cd] 'cafe',那么不論name是cafe、Cafe還是café上面的表達(dá)式都會返回YES。
  • 舉例說明
//創(chuàng)建 Person 類數(shù)組
    Person *p0 = [Person personName:@"ZhangSan" withAge:20 withSex:@"man"];
    Person *p1 = [Person personName:@"HanMeiMei" withAge:12 withSex:@"woman"];
    Person *p2 = [Person personName:@"LiLei" withAge:13 withSex:@"man"];
    Person *p3 = [Person personName:@"XiaoHua" withAge:13 withSex:@"woman"];
    NSArray *arr = @[p0, p1, p2, p3];
    pred = [NSPredicate predicateWithFormat:@"name LIKE '??an*'"];  //名字中第三,四位是 an.
    resultArr = [arr filteredArrayUsingPredicate:pred];
    NSLog(@"1?? : %@", resultArr);
    
    pred = [NSPredicate predicateWithFormat:@"name LIKE '*an*'"];   //包含 字符串模板an.
    resultArr = [arr filteredArrayUsingPredicate:pred];
    NSLog(@"2?? : %@", resultArr);
    
    pred = [NSPredicate predicateWithFormat:@"NOT (name CONTAINS 'ua')"];   // 不包含 ua.
    resultArr = [arr filteredArrayUsingPredicate:pred];
    NSLog(@"3?? : %@", resultArr);
    
    
    pred = [NSPredicate predicateWithFormat:@"name LIKE '*ua*'"];
    NSLog(@"判斷返回bool值 : %d", [pred evaluateWithObject:p3]);
//結(jié)果打印

"1?? : (
    "ZhangSan, 20, man"
)

2?? : (
   "ZhangSan, 20, man",
   "HanMeiMei, 12, woman"
)

3?? : (
    "ZhangSan, 20, man",
    "HanMeiMei, 12, woman",
    "LiLei, 13, man"
)

判斷返回bool值 : 1

  • MATCHES 舉例
 /** 謂詞匹配正則 */
    NSString *phoneStr = @"15242335566";
    NSLog(@"驗證 : %d", [self checkPhoneNumber:phoneStr]);
//結(jié)果打印
驗證 : 1
- (BOOL)checkPhoneNumber:(NSString *)phoneNumber
{
    NSString *regex = @"^[1][3-8]\\d{9}$";
    NSPredicate *pred = [NSPredicate predicateWithFormat:@"SELF MATCHES %@", regex];
    return [pred evaluateWithObject:phoneNumber];
    
   // 只有在正則表達(dá)式為^表達(dá)式$時才使用謂詞,而不是所有情況都使用, 具體說明, 自行查看文章末尾參考鏈接, 作者便不再展開說明
    
}
- 3. 集合運(yùn)算符
  • ANY、SOME:集合中任意一個元素滿足條件,就返回YES。
  • ALL:集合中所有元素都滿足條件,才返回YES。
  • NONE:集合中沒有任何元素滿足條件就返回YES。如:NONE person.age < 18,表示person集合中所有元素的age>=18時,才返回YES。
  • IN:等價于SQL語句中的IN運(yùn)算符,只有當(dāng)左邊表達(dá)式或值出現(xiàn)在右邊的集合中才會返回YES。我們通過一個例子來看一下
  • 舉例

 /** 從第二個數(shù)組中去除第一個數(shù)組中相同的元素 */
    NSArray *filterArray = @[@"ab", @"abc"];
    NSArray *array = @[@"a", @"ab", @"abc", @"abcd"];
    pred = [NSPredicate predicateWithFormat:@"NOT (SELF IN %@)", filterArray];
    NSLog(@"去除相同元素 :  %@", [array filteredArrayUsingPredicate:pred]);
//結(jié)果打印
去除相同元素 :  (
    a,
    abcd
)
- 4. 謂詞中使用占位符參數(shù)

首先如果我們想在謂詞表達(dá)式中使用變量,那么我們需要了解下列兩種占位符:

%K:用于動態(tài)傳入屬性名
%@:用于動態(tài)設(shè)置屬性值
其實(shí)相當(dāng)于變量名與變量值

>  除此之外,還可以在謂詞表達(dá)式中使用動態(tài)改變的屬性值,就像環(huán)境變量一樣

NSPredicate *pred = [NSPredicate predicateWithFormat:@"SELF CONTAINS $VALUE"];

>  上述表達(dá)式中,$VALUE是一個可以動態(tài)變化的值,它其實(shí)最后是在字典中的一個key,所以可以根據(jù)你的需要寫不同的值,但是必須有$開頭,隨著程序改變$VALUE這個謂詞表達(dá)式的比較條件就可以動態(tài)改變。
  • 舉例
//創(chuàng)建 Person 類數(shù)組
    Person *p0 = [Person personName:@"ZhangSan" withAge:20 withSex:@"man"];
    Person *p1 = [Person personName:@"HanMeiMei" withAge:12 withSex:@"woman"];
    Person *p2 = [Person personName:@"LiLei" withAge:13 withSex:@"man"];
    Person *p3 = [Person personName:@"XiaoHua" withAge:13 withSex:@"woman"];
    NSArray *arr = @[p0, p1, p2, p3];
    NSString *property = @"name";
    NSString *value = @"LiLei";
    //  該謂詞的作用是如果元素中property屬性含有值value時就取出放入新的數(shù)組內(nèi),這里是name包含LiLei
    pred = [NSPredicate predicateWithFormat:@"%K CONTAINS %@", property, value];
    NSArray *newArray = [arr filteredArrayUsingPredicate:pred];
    NSLog(@"newArray:%@", newArray);


    //  創(chuàng)建謂詞,屬性名改為age,要求這個age包含$VALUE字符串
    NSPredicate *predTemp = [NSPredicate predicateWithFormat:@"%K > $VALUE", @"age"];
    // 指定$SUBSTR的值為 12    這里注釋中的$SUBSTR改為$VALUE
    NSPredicate *pred1 = [predTemp predicateWithSubstitutionVariables:@{@"VALUE" : @12}];
    NSArray *newArray1 = [arr filteredArrayUsingPredicate:pred1];
    NSLog(@"newArray1:%@", newArray1);
    

    //  修改 $SUBSTR的值為13,  這里注釋中的SUBSTR改為$VALUE
    NSPredicate *pred2 = [predTemp predicateWithSubstitutionVariables:@{@"VALUE" : @13}];
    NSArray *newArray2 = [arr filteredArrayUsingPredicate:pred2];
    NSLog(@"newArray2:%@", newArray2);
//結(jié)果打印

 newArray:(
    "LiLei, 13, man"
)

newArray1:(
    "ZhangSan, 20, man",
    "LiLei, 13, man",
    "XiaoHua, 13, woman"
)

newArray2:(
    "ZhangSan, 20, man"
)
- 5. 謂詞中保留字
  • 保留字
    下列單詞都是保留字(不論大小寫)
    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
    注:雖然大小寫都可以,但是更推薦使用大寫來表示這些保留字

參考 : http://www.cocoachina.com/ios/20160111/14926.html

以上 !

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

推薦閱讀更多精彩內(nèi)容

  • 世事洞明皆學(xué)問,人情練達(dá)即文章。 在自己擅長的領(lǐng)域, 每個人都有超乎尋常的自尊心和自信心,但不要自負(fù)。不同的...
    和玨貓閱讀 796評論 3 4
  • 前言 在項目開發(fā)中遇見本地化搜索不是很高,當(dāng)有本地化搜索的需求時候?qū)SPredicate的語法都去網(wǎng)上搜索,然而...
    Ucself閱讀 2,350評論 0 2
  • 轉(zhuǎn)載自:http://www.cocoachina.com/ios/20160111/14926.html 1、大...
    一筆春秋閱讀 2,866評論 0 2
  • 鬧鐘一如既往5點(diǎn)響起,平時會迷迷糊糊地渴望賴床一會,今天異常清醒。睜開眼,看著黑暗的房間,放空自己,似乎什么也不想...
    許多多的后花園閱讀 328評論 1 0
  • 小書童:(搖著莫須有的扇子)現(xiàn)今在下行文類工作,需諸位多加配合…… 眾人:說人話!!! 小書童:我在做實(shí)習(xí)編輯,要...
    緋焱閱讀 285評論 0 1