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:]
如果对某个类,不允许使用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简述
KVO其实是一种观察者模式,利用它可以很容易实现视图组件和数据模型的分离,当数据模型的属性值改变之后作为监听器的视图组件就会被激发,激发时就会回调监听器自身。在ObjC中要实现KVO则必须实现NSKeyValueObServing协议,不过幸运的是NSObject已经实现了该协议,因此几乎所有的ObjC对象都可以使用KVO
在ObjC中使用KVO操作常用的方法如下:
注册指定Key路径的监听器:addObserver: forKeyPath: options: context:
删除指定Key路径的监听器:removeObserver: forKeyPath、removeObserver: 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应用。