我们一般是通过调用set方法或属性的点语法来直接更改对象的状态,即对象的属性值,比如[stu setAge:10]; stu.age = 9;
KVC,它是一种间接更改对象状态的方式,其实现方法是使用字符串来描述对象需要更改的属性。KVC中的基本调用包括valueForKey:和setValue:ForKey:,以字符串的形式向对象发送消息
#import <Foundation/Foundation.h> @interface Card : NSObject @property (nonatomic, copy) NSString *no; @end
#import <Foundation/Foundation.h> @class Card; @interface Person : NSObject @property (nonatomic, assign) int age; @property (nonatomic, copy) NSString *name; @property (nonatomic, retain) Card *card; @property (nonatomic, retain) NSArray *books; @end
valueForKey:
使用valueForKey:获取Student对象的name
NSString *name = [student valueForKey:@"name"];
valueForKey:首先查找以name或isName命名的getter方法,如果不存在getter方法,就在对象内部查找名为_name或name的实例变量
注意,下列语句是合法的
NSLog(@"age is %@", [student valueForKey:@"age"]);
NSLog中的%@输出一个对象,但age是一个int值,而不是对象,为什么会合理呢?原因如下:
使用valueForKey:时,KVC会自动将标量值(int、float、struct等)翻入NSNumber或NSValue中包装成一个对象,然后返回。因此,KVC有自动包装功能。
setValue:ForKey:
使用setValue:ForKey:设置Student对象的name
[student setValue:@"MJ" forKey:@"name"];
这个方法的工作方式和valueForKey:相同,首先查找setter方法,例如setName:,如果不存在setter方法,就在类中查找名为name或者_name的实例变量,然后为它赋值
使用setValue:ForKey:设置Student对象的age
[student setValue:[NSNumber numberWithInt:17] forKey:@"age"];
在设置一个标量值时,需要先将标量值包装成一个NSNumber或NSValue对象
批处理
KVC可以对对象进行批量更改
例如,同时获取Student的age和name
NSArray *keys = [NSArray arrayWithObjects:@"name", @"age", nil];
NSDictionary *dict = [student dictionaryWithValuesForKeys:keys];
同时设置Student的age和name
NSArray *keys = [NSArray arrayWithObjects:@"name", @"age", nil];
NSArray *values = [NSArray arrayWithObjects:@"MJ", [NSNumber numberWithInt:16], nil];
NSDictionary *dict = [NSDictionary dictionaryWithObjects:values forKeys:keys];
[student setValuesForKeysWithDictionary:dict];
键路径(key path)
除了通过键设置值外,KVC还支持键路径,像文件系统路径一样,其实就是属性链式访问
比如,利用键路径设置Student对象中Card对象的no
[student setValue:@"12345" forKeyPath:@"card.no"];
获取Student对象中Card对象的no
[student valueForKeyPath:@"card.no"];
数组的整体操作
如果向一个NSArray请求一个key,KVC会查询数组中的每个对象来查找这个key,然后将查询结果打包到另一个数组中并返回
例如,Student里面有很多Book对象
NSArray *names = [student.books valueForKeyPath:@"name"]; 或者
NSArray *names = [student valueForKeyPath:@"books.name"];
注意:不能在键路径中为数组添加索引,比如@"books[0].name"
键路径的运算符
在键路径中,可以引用一些运算符来进行一些运算,例如获取一组值的平均值、最小值、最大值或者总数
例如,计算Student中Book的总数
NSNumber *count = [student.books valueForKeyPath:@"@count"]; 或者
NSNumber *count = [student valueForKeyPath:@"books.@count"];
计算Student中所有Book的价钱(price)总和
NSNumber *sum = [student.books valueForKeyPath:@"@sum.price"]; 或者
NSNumber *sum = [student valueForKeyPath:@"[email protected]"];
找出Student中Book的所有不同价位(排除相同价位)
NSArray *prices = [student.books valueForKeyPath:@"@distinctUnionOfObjects.price"];
或者
NSArray *prices = [student valueForKeyPath:@"[email protected]"];
KVO是一种非常重要的机制,它允许监听对象的属性的变化
注册监听器
-(void)addObserver:(NSObject *)anObserver forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context
anObserver :监听器对象
keyPath :监听的属性
options :决定了当属性改变时,要传递什么数据给监听器
监听器需要实现监听方法
-(void) observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context
keyPath :监听的属性
object :谁的属性改变了
change :属性改变时传递过来的信息(取决于添加监听器时的options参数)
移除监听器
-(void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath
// // PersonObserver.m // 05-KVO // // Created by apple on 13-7-21. // #import "PersonObserver.h" @implementation PersonObserver #pragma mark 当监听的某个属性发生改变时调用 /* keyPath : 监听的属性名称 object : 监听的是哪个对象的属性 change : 属性发生的改变 context : 当初添加监听器时传入的参数 */ - (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context { //NSLog(@"%@-%@-%@", object, keyPath, context); NSLog(@"change=%@", change); } @end
// // main.m // 05-KVO // // Created by apple on 13-7-21. #import <Foundation/Foundation.h> #import "Person.h" #import "PersonObserver.h" int main(int argc, const char * argv[]) { @autoreleasepool { Person *p = [[Person alloc] init]; p.age = 10; p.name = @"jack"; PersonObserver *po = [[PersonObserver alloc] init]; int options = NSKeyValueObservingOptionOld | NSKeyValueObservingOptionNew; // 添加对象p的name属性监听器(observer) [p addObserver:po forKeyPath:@"name" options:options context:@"432432"]; p.name = @"Mike"; p.age = 11; // 删除监听器 [p removeObserver:po forKeyPath:@"name"]; p.name = @"rose"; } return 0; }