KVC与KVO的理解与应用

kvc简述

kvc即键值编码,在iOS中的应用主要体现在开发者通过key访问对象的属性或给对象的属性赋值。这样做最主要的好处是把访问和改变属性的动作放在了运行时,不需要再编译时确定

kvc常用的四个方法

(nullable id)valueForKey:(NSString *)key;                          //直接通过Key来取值

- (void)setValue:(nullable id)value forKey:(NSString *)key;          //通过Key来设值

- (nullable id)valueForKeyPath:(NSString *)keyPath;                  //通过KeyPath来取值

- (void)setValue:(nullable id)value forKeyPath:(NSString *)keyPath;  //通过KeyPath来设值

forKeyPath包含了forKey的功能,以后使用forKeyPath就可以了

forKeyPath中可以利用.运算符, 就可以一层一层往下查找对象的属性。而forKey是无法识别.运算符的。


kvc的具体实现原理


setValue:forKey:赋值原理如下:

去模型中查找有没有对应的setter方法:例如:setIcon方法,有就直接调用这个setter方法给模型这个属性赋值[self setIcon:dic[@"icon"]];

如果找不到setter方法,接着就会去寻找有没有icon属性,如果有,就直接访问模型中的icon属性,进行赋值,icon=dict[@"icon"];

如果找不到icon属性,接着又会去寻找_icon属性,如果有,直接进行赋值_icon=dict[@"icon"];如果都找不到就会报错:[setValue:forUndefinedKey:]

参考知识OC中属性self.a与_a访问的差别 

如果对某个类,不允许使用KVC,可以通过设置 accessInstanceVariablesDirectly 控制

KVC内部的实现

比如说如下的一行KVC的代码:

[site setValue:@"sitename" forKey:@"name"];

就会被编译器处理成:

SEL sel = sel_get_uid ("setValue:forKey:");

IMP method = objc_msg_lookup (site->isa,sel);

method(site, sel, @"sitename", @"name");

这下KVC内部的实现就很清楚的清楚了:一个对象在调用setValue的时候,(1)首先根据方法名找到运行方法的时候所需要的环境参数。(2)他会从自己isa指针结合环境参数,找到具体的方法实现的接口。(3)再直接查找得来的具体的方法实现。


kvo简述


摘自kenshincui 

KVO其实是一种观察者模式,利用它可以很容易实现视图组件和数据模型的分离,当数据模型的属性值改变之后作为监听器的视图组件就会被激发,激发时就会回调监听器自身。在ObjC中要实现KVO则必须实现NSKeyValueObServing协议,不过幸运的是NSObject已经实现了该协议,因此几乎所有的ObjC对象都可以使用KVO

在ObjC中使用KVO操作常用的方法如下:

注册指定Key路径的监听器:addObserver: forKeyPath: options:  context:

删除指定Key路径的监听器:removeObserver: forKeyPathremoveObserver: forKeyPath: context:

回调监听:observeValueForKeyPath: ofObject: change: context:

KVO的使用步骤也比较简单:

通过addObserver: forKeyPath: options: context:为被监听对象(它通常是数据模型)注册监听器

重写监听器的observeValueForKeyPath: ofObject: change: context:方法

由于我们还没有介绍过IOS的界面编程,这里我们还是在上面的例子基础上继续扩展,假设当我们的账户余额balance变动之后我们希望用户可以及时获得通知。那么此时Account就作为我们的被监听对象,需要Person为它注册监听(使用addObserver:

forKeyPath: options:

context:);而人员Person作为监听器需要重写它的observeValueForKeyPath: ofObject: change:

context:方法,当监听的余额发生改变后会回调监听器Person监听方法(observeValueForKeyPath: ofObject:

change: context:)。

。下面通过代码模拟上面的过程:

Account.h

/

//  Account.h

//  KVCAndKVO

//

//  Created by Kenshin Cui on 14-2-16.

//  Copyright (c) 2014年 Kenshin Cui. All rights reserved.

//#import@interfaceAccount : NSObject#pragmamark - 属性#pragmamark 余额

@property(nonatomic,assign)floatbalance;

@end

Account.m

//

//  Account.m

//  KVCAndKVO

//

//  Created by Kenshin Cui on 14-2-16.

//  Copyright (c) 2014年 Kenshin Cui. All rights reserved.

//#import"Account.h"@implementation Account

@end

Person.h

/

//  Person.h

//  KVCAndKVO

//

//  Created by Kenshin Cui on 14-2-16.

//  Copyright (c) 2014年 Kenshin Cui. All rights reserved.

//#import@classAccount;

@interfacePerson : NSObject{

@private

int_age;

}#pragmamark - 属性#pragmamark 姓名

@property(nonatomic,copy) NSString *name;#pragmamark 账户

@property(nonatomic,retain) Account *account;#pragmamark - 公共方法#pragmamark 显示人员信息

-(void)showMessage;

@end

Person.m

//

//  Person.m

//  KVCAndKVO

//

//  Created by Kenshin Cui on 14-2-16.

//  Copyright (c) 2014年 Kenshin Cui. All rights reserved.

//#import"Person.h"#import"Account.h"@implementation Person#pragmamark - 公共方法#pragmamark 显示人员信息

-(void)showMessage{

NSLog(@"name=%@,age=%d",_name,_age);

}#pragmamark 设置人员账户

-(void)setAccount:(Account *)account{

_account=account;//添加对Account的监听[self.account addObserver:self forKeyPath:@"balance"options:NSKeyValueObservingOptionNew context:nil];

}#pragmamark - 覆盖方法#pragmamark 重写observeValueForKeyPath方法,当账户余额变化后此处获得通知

-(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void*)context{if([keyPath isEqualToString:@"balance"]){//这里只处理balance属性NSLog(@"keyPath=%@,object=%@,newValue=%.2f,context=%@",keyPath,object,[[change objectForKey:@"new"] floatValue],context);

}

}#pragmamark 重写销毁方法

-(void)dealloc{

[self.account removeObserver:self forKeyPath:@"balance"];//移除监听

//[super dealloc];//注意启用了ARC,此处不需要调用}

@end

main.m

//

//  main.m

//  KVCAndKVO

//

//  Created by Kenshin Cui on 14-2-16.

//  Copyright (c) 2014年 Kenshin Cui. All rights reserved.

//#import#import"Person.h"#import"Account.h"intmain(intargc,const char* argv[]) {

@autoreleasepool {

Person *person1=[[Person alloc]init];

person1.name=@"Kenshin";

Account *account1=[[Account alloc]init];

account1.balance=100000000.0;

person1.account=account1;

account1.balance=200000000.0;//注意执行到这一步会触发监听器回调函数observeValueForKeyPath: ofObject: change: context:

//结果:keyPath=balance,object=,newValue=200000000.00,context=(null)}return0;

}

在上面的代码中我们在给人员分配账户时给账户的balance属性添加了监听,并且在监听回调方法中输出了监听到的信息,同时在对象销毁时移除监听,这就构成了一个典型的KVO应用。

你可能感兴趣的:(KVC与KVO的理解与应用)