KVC和KVO的学习
- KVC介绍
KVC的本质就是键值编码
定义:在对象创建完成之后,动态(牵扯到运行时)给对象的属性赋值
KVC 即是指NSKeyValueCoding,一个非正式的Protocol,提供一种机制来间接访问对象的属性,而不是通过调用setter,getter方法访问 - KVC的强大之处
<1> 访问私有变量
一个类中的私有变量,不能直接通过Setter,Getter方法访问,但是却可以通过KVC来访问
@interface Dog:NSObject{
@private
double height;
}
Dog *dog = [[Dog alloc]init];
[dog setValue:@12 forKey:@"height"];//给私有变量赋值
NSLog(@"dog's height is = %@",[dog valueForKey:@"height"]);//读取私有变量的值 ```
实际用例
利用KVC的这个特性,我们可以访问系统里的一些私有变量,例如在UIPageControl里面有两个私有变量
UIImage *_currentPageImage;
UIImage *_pageImage;
我们通过KVC来进去读取和赋值
UIPageControl *pageControl = [[UIPageControl alloc]init];
//设置值
[pageControl setValue:[UIImage imageNamed:@"XX"] forKeyPath:@"_currentPageImage"];
[pageControl setValue:[UIImage imageNamed:@"XX"] forKeyPath:@"_pageImage"];
//读取值
UIImage *currentImage = [pageControl valueForKey:@"_currentPageImage"];
UIImage *pageImage = [pageControl valueForKey:@"_pageImage"];
UIPageControl *pageControl = [[UIPageControl alloc]init];
//设置值
[pageControl setValue:[UIImage imageNamed:@"XX"] forKeyPath:@"_currentPageImage"];
[pageControl setValue:[UIImage imageNamed:@"XX"] forKeyPath:@"_pageImage"];
//读取值
UIImage *currentImage = [pageControl valueForKey:@"_currentPageImage"];
UIImage *pageImage = [pageControl valueForKey:@"_pageImage"];
使用KVC直接访问NSArray或者NSSet的属性值
NSArray *books= @[book1, book2, book3];
NSArray *names = [books valueForKeyPath:@"name"];
NSLog(@"%@", [books valueForKeyPath:@"@avg.price"]);//使用kvc直接打印出来书的平均价格
其中,book是一个书类,有一个属性是name,一个属性是price。
使用KVC将字典转化成模型
// 定义一个字典
NSDictionary *dict = @{
@"name" : @"jack",
@"money" : @"20.7",
};
// 创建模型
Person *p = [[Person alloc] init];
// 字典转模型
[p setValuesForKeysWithDictionary:dict];
NSLog(@"person's name is the %@",p.name);}
注意:
(1). key的值必须正确,如果拼写错误,会出现异常
(2). 当key的值是没有定义的,valueForUndefinedKey:这个方法会被调用,如果你自己写了这个方法,key的值出错就会调用到这里来
(3). 因为类key反复嵌套,所以有个keyPath的概念,keyPath就是用.号来把一个一个key链接起来,这样就可以根据这个路径访问下去
(4). NSArray/NSSet等都支持KVC
KVO的本质就是键值监听,他是基于KVC实现的关键技术之一,
定义:Key-Value Observing ,他提供一种机制,当指定的对象的属性被修改后,则对象会接收到通知
KVO实现原理是当XXX类第一次被监听时,系统会再运行时期动态的创建一个继承XXX的NSKVONOtifying_xxx类,在这个子类中,重写基类中任何被观察属性的setter方法,而没有被观察的属性则没有重新生成setter方法,在setter方法中实现真正的通知机制
- (void)setAge:(int)age{
[self willChangeValueForKey:@“age”];
[super setAge:age];
[self didChangeValueForKey:@"age"];//这个方法负责调用
-(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context;
}
###KVO,delegate,NSNotification及block的区别
KVO就是cocoa框架实现的观察者模式,一般同KVC搭配使用,通过KVO可以监测一个值的变化,比如View的高度变化,是一对多的关系,一个值的变化会通知所有的观察者
NSNotification是通知,也是一对多的使用场景,在某些情况下,KVO和NSNotification是一样的,都是状态变化之后告知对方。NSNotification的特点就是需要被观察者先主动发出通知,然后观察者注册监听之后再来进行响应,比起KVO多了发送通知的一步,但是其优点是监听不局限于属性的变化,还可以对多种多样的状态变化进行监听,监听范围广,使用也灵活
delegate是代理,就是我不想做的事情交给别人做,比如狗需要吃饭,就通过delegate通知主人,主人就会做饭,盛饭这些操作,这些狗都不需要关心,只需要调用delegate(代理人)就可以了,由其他类完成所需要的操作,所以delegate是一对一的关系
block是delegate的另一种形式,是函数式编程的一种形式,使用场景和delegate一样,相比delegate更灵活,实现上也更为直观。
KVO一般的使用场景是数据,需求是数据变化,比如股票价格变化,我们一般使用KVO(观察者模式).
delegate一般的使用场景是行为,需求是需要别人来帮我做一件事情,比如买卖股票,我们一般使用delegate
NSNOtification一般是进行全局通知,比如利好消息一出,通知大家去买入,delegate是强关联,就是委托和代理双方互相知道,你委托别人买股票你就需要知道经纪人,但是经纪人也要知道自己的顾客。Notifation是弱关联,利好消息发出,你不需要知道是谁发的也可以做出相应的反应,同理,发消息的人也不需要知道有谁接收,也能发出消息
##参考文献
--[http://www.mamicode.com/info-detail-515516.html]
--[http://blog.csdn.net/woaifen3344/article/details/48054729] 标哥的技术博客
网友评论