15、KVC、KVO和谓词

来源:互联网 发布:毒药法术升级数据 编辑:程序博客网 时间:2024/06/07 20:53

KVC的基本用法

键值编码的基本概念和用法

1.键-值编码是一个用于间接访问对象属性的机制,使用该机制不需要调用存取方 法和变量实例就可访问对象属性。
2.键-值编码方法在Objective-C非正式协议(类目)NSKeyValueCoding中被声 明,默认的实现方法由NSObject提供。
3.键-值编码支持带有对象值的属性,同时也支持纯数值类型和结构。非对象参数 和返回类型会被识别并自动封装/解封。

·设置和访问

键/值编码中的基本调用包括-valueForKey:和-setValue:forkey:这两个方 法,它们以字符串的形式向对象发送消息,字符串是我们关注属性的关键。

Book *book = [[Book alloc] init];NSMutableString *name = [[NSMutableString alloc] initWithFormat:@"jack"];[book setValue:name forKey:@"name"];NSLog(@"jack name : %@", [book valueForKey:@"name"]);

是否存在setter、getter方法,如果不存在,它将在内部查找名为_key或key的
实例变量。通过KVC,可以获取不存在getter方法的对象值,无需通过对象指针
直接访问。

这里我们需要注意,当我们通过setValue:forKey:设置对象的值,或通过valueForKey来获取对象的值时,如若对象的实例变量为基本数据类型时 (char、int、float、BOOL),我们需要对数据进行封装。

·路径

除了通过键设置值外,键/值编码还支持指定路径,像文件系统一样。用“点”号 隔开。

[book setValue:@"比尔" forKeyPath:@"author._name"]; [book valueForKeyPath:@"realtiveBooks.price"]

·一对多的关系

如果向NSArray请求一个键值,它实际上会查询数组中的每个对象来查找这个键 值,然后将查询结果打包到另一个数组中并返回给你。

NSArray *booksArray = [NSArray arrayWithObjects:book1, book2, nil];[book setValue:booksArray forKey:@"relativeBooks"];NSLog(@"books 2: %@", [book valueForKeyPath:@"relativeBooks.price"]);

·KVC的简单运算

此外,还可以应用一些字符做简单运算。sum、min、max、avg、count

NSNumber *count = [book valueForKeyPath:@"relativeBooks.@count"];NSLog(@"count : %@", count);NSNumber *sum = [book valueForKeyPath:@"relativeBooks.@sum._price"];NSLog(@"sum : %@", sum);NSNumber *avg = [book valueForKeyPath:@"relativeBooks.@avg._price"];NSLog(@"avg : %@", avg);NSNumber *min = [book valueForKeyPath:@"relativeBooks.@min._price"];NSLog(@"min : %@", min);NSNumber *max = [book valueForKeyPath:@"relativeBooks.@max._price"];NSLog(@"max : %@", max); 

例子

//Author.h#import <Foundation/Foundation.h>@interface Author : NSObject {@private    NSString *_name;}@end//Author.m#import "Author.h"@implementation Author@end//Book.h#import <Foundation/Foundation.h>#import "Author.h"@interface Book : NSObject {@private    NSString *_name;    Author *_author;    //书的作者    NSArray *_relativeBooks; //相关的书籍    float price;        //书的价格}@end//Book.m#import "Book.h"@implementation Book@end//main.h#import <Foundation/Foundation.h>#import "Book.h"int main(int argc, const char * argv[]){    @autoreleasepool {        Book *book = [[Book alloc] init];        Author *author = [[Author alloc] init];        [author setValue:@"jack" forKey:@"_name"];        [book setValue:[NSNumber numberWithFloat:12.6] forKey:@"price"];        NSNumber *price = [book valueForKeyPath:@"price"];        NSLog(@"%@",price);        //键值访问        [book setValue:@"ipad develper" forKey:@"name"];        [book setValue:author forKey:@"_author"];        //路径访问        [book setValue:@"tom" forKeyPath:@"author.name"];        NSString *authorName = [book valueForKeyPath:@"_author._name"];        NSLog(@"%@",authorName);        //一对多的关系        NSMutableArray *releBooks = [NSMutableArray arrayWithCapacity:3];        for (int i=0; i<3; i++) {            Book *book = [[Book alloc] init];            NSString *name = [NSString stringWithFormat:@"job_%d",i];            [book setValue:name forKey:@"_name"];            [book setValue:@(12+i) forKey:@"price"];            [releBooks addObject:book];            [book release];        }        [book setValue:releBooks forKey:@"_relativeBooks"];        NSArray *names = [book valueForKeyPath:@"_relativeBooks._name"];        NSLog(@"%@",names);        NSArray *names2 = [releBooks valueForKeyPath:@"_name"];        NSLog(@"%@",names2);        //运算, 运算的字段必须是数值类型NSNumber或者基本数据类型,计算的结果是NSNumber        //运算关键字sum、min、max、avg、count        NSNumber *sum = [book valueForKeyPath:@"_relativeBooks.@sum.price"];        NSLog(@"sum: %@",sum);    }    return 0;}

KVO的基本概念

·基本概念

键-值观察是一种使对象获取其他对象的特定属性变化的通知机制。

控制器层的绑定技术就是严重依赖键-值观察获得模型层和控制器层的变化通知 的。对于不依赖控制器层类的应用程序,键-值观察提供了一种简化的方法来实现 检查器并更新用户界面值。

与 NSNotification不同,键-值观察中并没有所谓的中心对象来为所有观察者提 供变化通知。取而代之地,当有变化发生时,通知被直接发送至处于观察状态的对 象。NSObject提供这种基础的键-值观察实现方法,你几乎不用重写该方法。

你可以观察任意对象属性,包括简单属性,对一或是对多关系。对多关系的观察 者将会被告知发生变化的类型-也就是任意发生变化的对象。

键-值观察为所有对象提供自动观察兼容性。你可以通过禁用自动观察通知并实现 手动通知来筛选通知。

·注册观察者

为了正确接收属性的变更通知,观察对象必须首先发送一个 addObserver:forKeyPath:options:context:消息至被观察对象,用以传送观察 对象和需要观察的属性的关键路径,以便与其注册。选项参数指定了发送变更通知 时提供给观察者的信息。使用NSKeyValueObservingOptionOld选项可以将初始 对象值以变更字典中的一个项的形式提供给观察者。指定 NSKeyValueObservingOptionNew选项可以将新的值以一个项的形式添加至变更字典。你可以使用逐位“|”这两个常量来指定接收上述两种类型的值。

·示例:

[child addObserver:self forKeyPath:@"tired" options:NSKeyValueObservingOptionOld|NSKeyValueObservingOptionNew context:nil];

·接受变更通知

当监听的属性发生变动时,观察者收到一 observeValueForKeyPath:ofObject:change:context:消息, 观察者必须实现这 一方法。触发观察通知的对象和键路径、包含变更细节的字典,以及观察者注册时提交的上下文指针均被提交给观察者。

·示例:

 - (void)observeValueForKeyPath:(NSString *)keyPath                      ofObject:(id)object                        change:(NSDictionary *)change                       context:(void *)c{    if ([keyPath isEqual:@"key"]) {        NSLog(@"");    } }

·移除观察者身份

你可以发送一条指定观察方对象和键路径的removeObserver:forKeyPath:消息 至被观察的对象,来移除一个键-值观察者。(当我们达到目的时)

·示例:

- (void)dealloc {    [_child removeObserver:self forKeyPath:@"happyVal"];    [_child release];    [super dealloc];}

谓词的基本概念

·概念

cocoa中提供了NSPredicate类,指定过滤器的条件。将符合条件的对象保留下来。

·创建谓词

/**** 设置谓词条件 ****/NSPredicate *predicate = [NSPredicate predicateWithFormat:@"age <= 28"]; for (Person *person in array) {    // 表示指定的对象是否满足谓词条件    if ([predicate evaluateWithObject:person]) {      //NSLog(@"person name : %@", person.name);    } }
/**** 返回个符合谓词条件的数组 ****/NSArray *newArray = [array filteredArrayUsingPredicate:predicate];for (Person *person in newArray) {    //NSLog(@"person name : %@", [person valueForKey:@"_name"]);}
 /**** 格式占位符号 ****/NSPredicate *pre = [NSPredicate predicateWithFormat:@" age <= %d", 30]; NSArray *array2 = [array filteredArrayUsingPredicate:pre];for (Person *person in array2) {     //NSLog(@"person name 2 %@", [person      valueForKey:@"_name"]);}

·逻辑运算符

/**** 运算符号 的加入 谓词不区分大小 && AND || OR ****/NSPredicate *pre3 = [NSPredicate predicateWithFormat:@"name > 'bruse' && age < %d", 30];NSArray *array4 = [array filteredArrayUsingPredicate:pre3];

·IN

/**** 关键字 注意字符串 定要添加'' ****/NSPredicate *pre4 = [NSPredicate predicateWithFormat:@"self.name IN {'rose', 'bruse'}"];NSArray *array5 = [array filteredArrayUsingPredicate:pre4]; NSLog(@"person name : %@", [array5 valueForKey:@"_name"]);

·以**开始——BEGINSWITH

/**** BEGINSWITH 检查某个字是否以**开头 ****/NSPredicate *pre5 = [NSPredicate predicateWithFormat:@"self.name BEGINSWITH 'J'"];NSArray *array6 = [array filteredArrayUsingPredicate:pre5]; NSLog(@"person name : %@", [array6 valueForKey:@"name"]);

·以**结束——ENDSWITH

/**** ENDSWITH 检查某个字符是以**结尾 ****/NSPredicate *pre6 = [NSPredicate predicateWithFormat:@"self.name endswith 'e'"];NSArray *array7 = [array filteredArrayUsingPredicate:pre6]; NSLog(@"array7 : %@", [array7 valueForKey:@"name"]);

·包含——CONTAINS

/**** CONTAINS 检查包含某个字符 ****/NSPredicate *pre8 = [NSPredicate predicateWithFormat:@"self.name CONTAINS '小'"];NSArray *array8 = [array filteredArrayUsingPredicate:pre8]; NSLog(@"array8 : %@", [array8 valueForKey:@"name"]);

·模糊查询——Like

/**** Like 检查包含某个字符 ****/NSString *s = [NSString stringWithFormat:@"name like '*%@*'",@"a"]; NSPredicate *predicate = [NSPredicate predicateWithFormat:s]; NSLog(@"array8 : %@", [array8 valueForKey:@"name"]);
0 0