一、谓词的基本概念与使用

1、谓词(NSPredicate)用于定义一个逻辑条件,通过该条件可执行搜索或内存中的过滤操作。上一篇文章中介绍的集合都提供了使用谓词对集合进行过滤的方法。OC中的谓词操作是针对于数组类型的,他就好比数据库中的查询操作,数据源就是数组,这样的好处是我们不需要编写很多代码就可以去操作数组,同时也起到过滤的作用,我们可以编写简单的谓词语句,就可以从数组中过滤出我们想要的数据。非常方便。在Java中是没有这种技术的,但是有开源的框架已经实现了此功能。

2、创建谓词之后,如果谓词中没有占位符,则可以直接使用NSPredicate的evaluateWithObject:方法计算谓词的结果,该结果总是一个BOOL值;

 #import <Foundation/Foundation.h>
#import "FKUser.h" int main(int argc , char * argv[])
{
@autoreleasepool{
// 创建谓词,要求name的以s开头
NSPredicate* pred = [NSPredicate predicateWithFormat:
@"name like 's*'"];
FKUser* user1 = [[FKUser alloc] initWithName:@"sun"
pass:@""];
// 对user1对象使用谓词执行判断。
BOOL result1 = [pred evaluateWithObject:user1];
NSLog(@"user1的name是否以s开头:%d", result1);
FKUser* user2 = [[FKUser alloc] initWithName:@"bai"
pass:@""];
// 对user1对象使用谓词执行判断。
BOOL result2 = [pred evaluateWithObject:user2];
NSLog(@"user2的name是否以s开头:%d", result2);
}
}

NSPredicate示例

3、谓词本身就代表了一个逻辑条件,计算谓词的结果就返回了BOOL类型的值,谓词一个常用的功能就说对集合进行过滤。当程序使用谓词对集合元素进行过滤时,程序会自动遍历集合元素,并根据集合元素计算谓词的值,只有根据某个集合元素计算谓词并返回YES时,该集合元素才会被保留下来。

  NSArray提供了如下方法使用谓词来过滤集合:

  • -(NSArray *)filteredArrayUsingPredicate:(NSPredicate *)predicate::使用制定谓词过滤NSArray集合,返回集合中符合谓词条件的元素组成新集合

  NSMutableArray提供了如下方法使用谓词来过滤集合:

  • -(void)filteredUsingPredicate:(NSPredicate *)predicate::使用制定谓词过滤NSMutableArray集合,剔除该集合中不符合谓词条件的元素

  NSSet提供如下方法使用谓词进行过滤集合:

  • -(NSSet *)filteredSetUsingPredicate:(NSPredicate *)predicate::使用制定谓词过滤NSArray集合,返回集合中符合谓词条件的元素组成新集合

 NSMutableSet提供了如下方法使用谓词来过滤集合:

  • -(void)filteredUsingPredicate:(NSPredicate *)predicate::使用制定谓词过滤NSMutableSet集合,剔除该集合中不符合谓词条件的元素
 #import <Foundation/Foundation.h>
#import "FKUser.h" int main(int argc , char * argv[])
{
@autoreleasepool{
NSMutableArray* array = [NSMutableArray
arrayWithObjects: [NSNumber numberWithInt:],
[NSNumber numberWithInt:],
[NSNumber numberWithInt:],
[NSNumber numberWithInt:],
[NSNumber numberWithInt:],
[NSNumber numberWithInt:],nil];
// 创建谓词,要求该对象自身的值大于50
NSPredicate* pred1 = [NSPredicate predicateWithFormat:
@"SELF > 50"];
// 使用谓词执行过滤,过滤后只剩下值大于50的集合元素
[array filterUsingPredicate:pred1];
NSLog(@"值大于50的元素:%@" , array);
NSSet* set = [NSSet setWithObjects:
[[FKUser alloc] initWithName:@"孙悟空"
pass:@""],
[[FKUser alloc] initWithName:@"金角大王"
pass:@""],
[[FKUser alloc] initWithName:@"猪八戒"
pass:@""],
[[FKUser alloc] initWithName:@"太上老君"
pass:@""],
[[FKUser alloc] initWithName:@"银角大王"
pass:@""], nil];
// 创建谓词,要求该对象的name值中包含'大王'
NSPredicate* pred2 = [NSPredicate predicateWithFormat:
@"name CONTAINS '大王'"];
// 执行过滤,过滤后集合只剩下两个元素
NSSet* newSet = [set filteredSetUsingPredicate:pred2];
NSLog(@"%@" , newSet);
}
}

NSPredicate过滤集合

4、在前面的程序中使用谓词的对象总是固定的,总是用指定对象的固定属性与固定的值进行比较,如果符合条件,则返回YES,否则返回NO。有时候我们需要在谓词表达式中使用变量,则可以考虑在谓词表达式中使用占位符参数,在谓词表达式中支持以下两个占位符参数:

  • %K:该占位符用于动态传入属性名。
  • %@:该占位符用于动态设置属性值。取决于要设置属性值的类型,该占位符也可以改成%d、%g等占位符

  如果谓词中由占位符参数,则需要经过两步来计算谓词的结果:

  1. 调用predicateWithSubstitutionVariables:方法为占位符参数设置参数值,该方法返回一个可用的NSPredicate对象
  2. 执行谓词的evaluateWithObject:方法计算谓词的返回结果
     #import <Foundation/Foundation.h>
    #import "FKUser.h" int main(int argc , char * argv[])
    {
    @autoreleasepool{
    NSSet* set = [NSSet setWithObjects:
    [[FKUser alloc] initWithName:@"孙悟空"
    pass:@""],
    [[FKUser alloc] initWithName:@"金角大王"
    pass:@""],
    [[FKUser alloc] initWithName:@"猪八戒"
    pass:@""],
    [[FKUser alloc] initWithName:@"太上老君"
    pass:@""],
    [[FKUser alloc] initWithName:@"银角大王"
    pass:@""], nil];
    NSString* propPath = @"name";
    NSString* value = @"大王";
    // 创建谓词,该谓词中包含了2个占位符
    // 后面的2个变量用于为占位符设置参数值,因此实际上相当于创建了谓词表达式 "name CONTAINS '大王'"
    NSPredicate* pred = [NSPredicate predicateWithFormat:
    @"%K CONTAINS %@" , propPath , value];
    // 执行过滤,过滤后集合只剩下两个元素
    NSSet* newSet = [set filteredSetUsingPredicate:pred];
    NSLog(@"%@" , newSet);
    // 创建谓词,该谓词表达式中使用%K占位符,该占位符使用pass代替,所以该代码相当于创建谓词表达式 "pass CONTAINS $SUBSTR"
    // $SUBSTR相当于一个变量,需要我们调用时为它设置值
    NSPredicate* predTemplate = [NSPredicate predicateWithFormat:
    @"%K CONTAINS $SUBSTR" , @"pass"];
    // 使用NDDictionary指定SUBSTR的值为'43',这就相当于创建了谓词表达式"pass CONTAINS '43'"
    NSPredicate* pred1 = [predTemplate
    predicateWithSubstitutionVariables:
    [NSDictionary dictionaryWithObjectsAndKeys:
    @"" , @"SUBSTR", nil]];
    // 执行过滤,过滤后集合只剩下两个元素
    NSSet* newSet1 = [set filteredSetUsingPredicate:pred1];
    NSLog(@"%@" , newSet1);
    // 使用NDDictionary指定SUBSTR的值为'59'
    NSPredicate* pred2 = [predTemplate
    predicateWithSubstitutionVariables:
    [NSDictionary dictionaryWithObjectsAndKeys:
    @"" , @"SUBSTR", nil]];
    // 执行过滤,过滤后集合只剩下两个元素
    NSSet* newSet2 = [set filteredSetUsingPredicate:pred2];
    NSLog(@"%@" , newSet2);
    }
    }

    NSPredicate使用占位符

二、谓词表达式语法

1、基本比较运算符:=、==(是否相等);>=、=>(左边是否大于或等于右边);<=、=<(左边是否小于或等于右边);>(左边是否大于右边);<(左边是否小于右边);!=、<>(不等于);BETWEEN(必须满足表达式 BETWEEN{下限,上限} 的格式,在范围内左右取等号)。

2、基本逻辑运算符:

  • AND、&&:逻辑与
  • OR、||:逻辑或
  • NOT、!:逻辑非

3、字符串比较运算符:

  • BEGINSWITH:检查某个字符串是否以指定的某个子串开头
  • ENDSWITH:检查某个字符串是否以指定的某个子串结尾
  • CONTAINS:检查某个字符串是否包含指定的某个子串
  • LIKE:检查某个字符串是否匹配指定的字符串模板
  • MATCHES:检查某个字符串是否匹配指定的正则表达式

  上述方法都是区分大小写的,如果要不区分大小写和重音符号,则可以在运算符后面使用[c]、[d]选项,其中[c]指定不区分大小写,[d]指定不区分重音符号。eg:“name BEGINSWITH[c][d] fkja”,则不管name值是FKJava还是fkjava都将返回YES。

4、操作集合的运算符:

  • ANY、SOME:指定只要集合中任意一个元素满足条件,即可返回YES。
  • ALL:指定所有元素满足才返回YES。
  • NONE:指定没有任何元素满足条件才返回YES。
  • IN:只有当左边的表达式或值出现在右边的集合中才会返回YES。
  • array[index]:返回array数组中索引为index处的元素
  • array[FIRST]:返回array数组中第一个元素
  • array[LAST]:返回array数组中最后一个元素
  • array[SIZE]:返回array数组中元素的个数。

5、直接量:

  在谓词表达式中,使用双引号和单引号的效果是一样的。但是单引号只能用单引号结束,不能混用。

  • FALSE、NO:逻辑假
  • TRUE、YES:逻辑真
  • NULL、NIL:代表一个空值
  • SELF:代表正在被判断的对象
  • "text"或'text':代表字符串
  • 数组:数组元素用英文字符隔开。eg:{'keli','zhangsan','lisi','wangwu'}
  • 数值直接量:包括整数、小数、科学计数法
  • 十六进制数:以0x开头
  • 八进制数:以0o开头
  • 二进制数:以0b开头

OC学习14——谓词的更多相关文章

  1. OC学习篇之---谓词(NSPredicate)

    在前一篇文章中我们介绍了OC中一个重要技术通知:http://blog.csdn.net/jiangwei0910410003/article/details/41923401,今天我们在来看一下OC ...

  2. OC学习篇之---总结和学习目录

    今天终于把OC的基础知识学习完了,但是这些知识只是最基础的,还有很多高级知识,这个可能需要后面慢慢的去学习才能体会到.下面就是这次学习OC的目录教程,如果大家发现有什么不正确的地方,请指正,小弟是新生 ...

  3. OC学习总结之面向对象和类

    OC学习总结之面向对象和类   Objective-c是c语言的母集合,它的原意就是在原始的c语言的主体上加入面向对象的特性.1.面向对象和面向过程  面向对象和面向过程是编程的两种思考方式.面向对象 ...

  4. IOS开发-OC学习-常用功能代码片段整理

    IOS开发-OC学习-常用功能代码片段整理 IOS开发中会频繁用到一些代码段,用来实现一些固定的功能.比如在文本框中输入完后要让键盘收回,这个需要用一个简单的让文本框失去第一响应者的身份来完成.或者是 ...

  5. OC学习16——对象归档

    转载自  OC学习篇之---归档和解挡 OC中的归档就是将对象写入到一个文件中,Java中的ObjectInputStream和ObjectOutputStream来进行操作的.当然在操作的这些对象都 ...

  6. OC学习11——循环引用与@class

    转载自 OC学习篇之---@class关键字的作用以及#include和#import的区别 一.#import和#include的区别 当我们在代码中使用两次#include的时候会报错:因为#in ...

  7. OC学习9——反射机制

    1.OC提供了3种编程方式与运行环境进行交互: 直接通过OC的源代码:这是最常见的方式,开发人员只是编写OC源代码,而运行环境负责在后台工作. 通过NSObject类中定义的方法进行动态编程:因为绝大 ...

  8. OC学习6——面相对象的三大特性

    我们在学习Java的时候都知道,类有三大特性:继承,封装,多态,这也是面向对象的三大特征.OC学习篇之---类的三大特性(封装,继承,多态) 1.封装(Encapsulation)是指将对象的状态信息 ...

  9. OC学习5——类和对象

    1.OC是在C语言基础上进行扩展得到的一门面向对象的程序设计语言,它也提供了定义类.成员变量和方法的基本功能.类可以被认为是一种自定义的数据类型,使用它可以定义变量,所有使用类定义的变量都是指针类型的 ...

随机推荐

  1. 深入理解js中的apply、call、bind

    概述 js中的apply,call都是为了改变某个函数运行时的上下文环境而存在的,即改变函数内部的this指向. apply() apply 方法传入两个参数:一个是作为函数上下文的对象,另外一个是作 ...

  2. 做技术,有没有必要参加IT培训

    近几年,IT培训机构可谓是琳琅满目,稂莠不齐.培训Java的,培训PHP的,培训大数据的等等吧,不一而足. 自己也算是IT技术圈子待了好多年了,面试过一些机构培训出来的学生,也有几个好哥们在培训机构做 ...

  3. Grafana+Prometheus系统监控之webhook

    概述 Webhook是一个API概念,并且变得越来越流行.我们能用事件描述的事物越多,webhook的作用范围也就越大.Webhook作为一个轻量的事件处理应用,正变得越来越有用. 准确的说webho ...

  4. CDH5.11..0安装

    1.参考: http://www.cnblogs.com/codedevelop/p/6762555.html grant all privileges on *.* to 'root'@'hostn ...

  5. java版Web Socket,实现消息推送

    # web socket是什么? WebSocket协议是基于TCP的一种新的网络协议. 它实现了浏览器与服务器全双工(full-duplex)通信,允许服务器主动发送信息给客户端. ## 用途 实时 ...

  6. 熵(Entropy),交叉熵(Cross-Entropy),KL-松散度(KL Divergence)

    1.介绍: 当我们开发一个分类模型的时候,我们的目标是把输入映射到预测的概率上,当我们训练模型的时候就不停地调整参数使得我们预测出来的概率和真是的概率更加接近. 这篇文章我们关注在我们的模型假设这些类 ...

  7. php代码中的细节问题

    本次主要谈及工作中关于注销功能中的路径设置及session的清除问题之前的调试一直忽略了session的功能,以至于每次使用__ROOT__/index.php/home/Student/logout ...

  8. django框架中的中间件

    什么是中间件 中间件就是在url进入路由之前进行检测的一个类 也就是说,每一个请求都是先通过中间件中的 process_request 函数,这个函数返回 None 或者 HttpResponse 对 ...

  9. NTP时间同步 服务端 客户端 自动化安装配置

    NTP时间同步 服务端 客户端 自动化安装配置 原创内容 http://www.cnblogs.com/elvi/p/7657994.html #!/bin/sh #运行环境 centos6.cent ...

  10. 51Nod 1084 矩阵取数问题 V2 双线程DP 滚动数组优化

    基准时间限制:2 秒 空间限制:131072 KB  一个M*N矩阵中有不同的正整数,经过这个格子,就能获得相应价值的奖励,先从左上走到右下,再从右下走到左上.第1遍时只能向下和向右走,第2遍时只能向 ...