iOS中谓词(NSPredicate)的基本入门使用教程

前言

首先,我们需要知道何谓谓词,让我们看看官方的解释:

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

  • >=,=>:判断左边表达式的值是否大于或等于右边表达式的值
  • <=,=<:判断左边表达式的值是否小于或等于右边表达式的值
  • >:判断左边表达式的值是否大于右边表达式的值
  • <:判断左边表达式的值是否小于右边表达式的值
  • !=、<>:判断两个表达式是否不相等
  • BETWEEN:BETWEEN表达式必须满足表达式 BETWEEN {下限,上限}的格式,要求该表达式必须大于或等于下限,并小于或等于上限
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
)

  • OR、||:逻辑或,要求其中一个表达式为YES时,结果就是YES
  • NOT、 !:逻辑非,对原有的表达式取反

3.字符串比较运算符

  • BEGINSWITH:检查某个字符串是否以指定的字符串开头(如判断字符串是否以a开头:BEGINSWITH 'a')
  • ENDSWITH:检查某个字符串是否以指定的字符串结尾
  • CONTAINS:检查某个字符串是否包含指定的字符串
  • LIKE:检查某个字符串是否匹配指定的字符串模板。其之后可以跟?代表一个字符和*代表任意多个字符两个通配符。比如"name LIKE '*ac*'",这表示name的值中包含ac则返回YES;"name LIKE '?ac*'",表示name的第2、3个字符为ac时返回YES。
  • MATCHES:检查某个字符串是否匹配指定的正则表达式。虽然正则表达式的执行效率是最低的,但其功能是最强大的,也是我们最常用的。

注:字符串比较都是区分大小写和重音符号的。如:café和cafe是不一样的,Cafe和cafe也是不一样的。如果希望字符串比较运算不区分大小写和重音符号,请在这些运算符后使用[c],[d]选项。其中[c]是不区分大小写,[d]是不区分重音符号,其写在字符串比较运算符之后,比如:name LIKE[cd] 'cafe',那么不论name是cafe、Cafe还是café上面的表达式都会返回YES。

4.集合运算符

  • ANY、SOME:集合中任意一个元素满足条件,就返回YES。
  • ALL:集合中所有元素都满足条件,才返回YES。
  • NONE:集合中没有任何元素满足条件就返回YES。如:NONE person.age < 18,表示person集合中所有元素的age>=18时,才返回YES。
  • IN:等价于SQL语句中的IN运算符,只有当左边表达式或值出现在右边的集合中才会返回YES。我们通过一个例子来看一下
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
)

  • array[index]:返回array数组中index索引处的元素
  • array[FIRST]:返回array数组中第一个元素
  • array[LAST]:返回array数组中最后一个元素
  • array[SIZE]:返回array数组中元素的个数

5.直接量

在谓词表达式中可以使用如下直接量

  • FALSE、NO:代表逻辑假
  • TRUE、YES:代表逻辑真
  • NULL、NIL:代表空值
  • SELF:代表正在被判断的对象自身
  • "string"或'string':代表字符串
  • 数组:和c中的写法相同,如:{'one', 'two', 'three'}。
  • 数值:包括证书、小数和科学计数法表示的形式
  • 十六进制数:0x开头的数字
  • 八进制:0o开头的数字
  • 二进制:0b开头的数字

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提供了如下方法使用谓词来过滤集合
    - (NSArray<ObjectType> *)filteredArrayUsingPredicate:(NSPredicate *)predicate:使用指定的谓词过滤NSArray集合,返回符合条件的元素组成的新集合
  • NSMutableArray提供了如下方法使用谓词来过滤集合
    - (void)filterUsingPredicate:(NSPredicate *)predicate:使用指定的谓词过滤NSMutableArray,剔除集合中不符合条件的元素
  • NSSet提供了如下方法使用谓词来过滤集合
    - (NSSet<ObjectType> *)filteredSetUsingPredicate:(NSPredicate *)predicate NS_AVAILABLE(10_5, 3_0) :作用同NSArray中的方法
  • NSMutableSet提供了如下方法使用谓词来过滤集合
    - (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循环等等。可以看出NSPredicate的出现为我们节省了大量的时间和精力。

3.在谓词中使用占位符参数

我们上面所有的例子中谓词总是固定的,然而我们在现实中处理变量时决定了谓词应该是可变的。下面我们来看看如果让谓词变化起来。

首先如果我们想在谓词表达式中使用变量,那么我们需要了解下列两种占位符

  • %K:用于动态传入属性名
  • %@:用于动态设置属性值

其实相当于变量名与变量值

除此之外,还可以在谓词表达式中使用动态改变的属性值,就像环境变量一样

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就到到此介绍完毕。

因为本人水平有限,如果有什么好的建议请联系我。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对我们的支持。

(0)

相关推荐

  • iOS中NSPredicate谓词的使用

    前言 NSPredicate 谓词, 其实 NSPredicate 有时候完全可以被 [for...][if..else] 替换来实现功能, 但会造成代码不够优雅, 一堆的判断循环 有时一句 NSPredicate 便可实现. 今天作者就聊一聊谓词的使用. 一 NSPredicate 理解说明 NSPredicate : 中文直译谓词, 用来定义逻辑条件约束的搜索 或 内存中的过滤. 如同语法中的谓词, 如 [3 大于 2]中"大于"就是一个就是谓词. 简单点说 它是逻辑判断, 如同过

  • IOS中NSPredicate和NSRegularExpression校验正则表达式区别

    在代码开发过程中,我们经常需要用来校验邮箱.手机号等等,这个时候就需要用到正则表达式.在iOS开发中,能用来做正则校验的有两个 NSPredicate 和 NSRegularExpression . NSPredicate NSPredicate 能用来简单做正则校验,但是它的问题是存在校验不出来的情况. //NSString+RegEx.h #import <Foundation/Foundation.h> @interface NSString (RegEx) #pragma mark -

  • 30分钟快速带你理解iOS中的谓词NSPredicate

    一.引言 在现代汉语的解释中,谓词是用来描述或判断客体性质.特征或者客体之间关系的词项.通俗的说,它是描述事物属性的.在iOS开发Cocoa框架中,有提供NSPredicate类,这个类通常也被成为谓词类,其主要的作用是在Cocoa中帮助查询和检索,但是需要注意,实质上谓词并不是提供查询和检索的支持,它是一种描述查询检索条件的方式,就像更加标准通用的正则表达式一样. NSPredicate提供的谓词可以分为两类:比较谓词和复合谓词. 比较谓词:比较谓词通过使用比较运算符来描述所符合条件的属性状态

  • iOS中谓词(NSPredicate)的基本入门使用教程

    前言 首先,我们需要知道何谓谓词,让我们看看官方的解释: The NSPredicate class is used to define logical conditions used to constrain a search either for a fetch or for in-memory filtering. NSPredicate类是用来定义逻辑条件约束的获取或内存中的过滤搜索. 可以使用谓词来表示逻辑条件,用于描述对象持久性存储在内存中的对象过滤.其实意思就是:我是一个过滤器,不

  • C#中char和string的入门使用教程

    char 字符 char代表一个Unicode字符,它是System.Char的别名 char someChar = 'a';//定义了一个字符 char newLine= '\n';//这是一个换行符 System.Char定义了一组静态方法: ToUpper 将指定的字符转换为等效的大写形式 ToLower 将指定的字符转换为等效的小写形式 IsWhiteSpace 判断指定的字符是否为空白字符 -- 例子: Console.WriteLine(char.ToUpper('c'));//输出

  • iOS中最全的各种定时器使用教程

    前言 相信一说到定时器, 我们使用最多的就是NSTimer 和 GCD 了, 还有另外一个高级的定时器 CADisplayLink;,下面将给大家详细介绍关于iOS定时器使用的相关内容,话不多说了,来一起看看详细的介绍吧. 一. NSTimer NSTimer的初始化方法有以下几种: 会自动启动, 并加入 MainRunloop 的 NSDefaultRunLoopMode 中, 注意: 这里的自动启动, 并不是马上就会启动, 而是会延迟大概一个interval的时间: + (NSTimer *

  • iOS中UIAlertView警告框组件的使用教程

    1. 最简单的用法 初始化方法: 复制代码 代码如下: - (instancetype)initWithTitle:(NSString *)title message:(NSString *)message delegate:(id /*<UIAlertViewDelegate>*/)delegate cancelButtonTitle:(NSString *)cancelButtonTitle otherButtonTitles:(NSString *)otherButtonTitles,

  • C语言中的结构体的入门学习教程

    C语言中数组允许定义类型的变量,可容纳相同类型的多个数据项,但结构体在C语言编程中,它允许定义不同种类的数据项可供其他用户定义的数据类型. 结构是用来代表一个记录,假设要跟踪图书馆的书籍.可能要跟踪有关每本书以下属性: Title - 标题 Author - 作者 Subject - 科目 Book ID - 编号 定义结构体 定义一个结构体,必须使用结构体的struct语句.该struct语句定义了一个新的数据类型,程序不止一个成员.struct语句的格式是这样的: struct [struc

  • C++中sort函数的基础入门使用教程

    前言 STL主要包含容器,迭代器,算法三块内容,用户可以对容器进行一系列的操作,比如遍历和计算,而STL提供的迭代器和容器完美地提供了这样的接口.其中std::vector是最常用的容器之一,vector是一个模板类,定义在命名空间namespace下,使用vector需要在包含相关头文件.今天主要讲解对vector的排序的使用. sort类函数: 函数名 功能描述 sort 对给定区间所有元素进行排序 stable_sort 对给定区间所有元素进行稳定排序 partial_sort 对给定区间

  • iOS中UIAlertView3秒后消失的两种实现方法

    一,效果图. 二,代码. - (void)viewDidLoad { [super viewDidLoad]; // Do any additional setup after loading the view, typically from a nib. UIAlertView* alert = [[UIAlertView alloc]initWithTitle:nil message:@"此信息3秒后消失" delegate:nil cancelButtonTitle:nil ot

  • iOS中多线程的入门使用教程(Swift)

    目录 一.iOS的三种多线程技术 二.基本使用 NSThread的基本使用 NSOperation的基本使用 GCD的基本使用 总结 一.iOS的三种多线程技术 1.NSThread –优点:NSThread 比其他两个轻量级,使用简单 –缺点:需要自己管理线程的生命周期.线程同步.加锁.睡眠以及唤醒等.线程同步对数据的加锁会有一定的系统开销 2.NSOperation 不需要关心线程管理,数据同步的事情,可以把精力放在自己需要执行的操作上 3.GCD 基于C语言的 二.基本使用 NSThrea

随机推荐