前言
KVC/KVO在日常开发中也是经常会使用到,但是还是回到那句话,往往我们在使用一种技术时,却不知道实现原理,以及会忽略一些使用时需要注意的地方。这篇文章会对KVC/KVO做一些原理性的说明,至于使用方式,大家自行度娘,网上还是挺多了,这里就不浪费ctrl+c了。希望文章对大家有帮助,同时欢迎大家指正、交流。
这篇主要说说KVO,关于KVC的相关请看这篇文章:iOS开发技巧系列---详解KVC(我告诉你KVC的一切),写的非常详细。
简介
KVO:Key-Value Observing(键值观察)
观察者设计模式
的一种实现,另外一种是:通知
(不在本文主题内)。KVO 提供一种机制,指定一个被观察对象(例如 A 类),当对象某个属性(例如 A 中的name属性)发生更改时,对象会获得通知,可作出相应处理。
那么KVO的实现原理是怎样的呢?
我们需要知道的是KVO是基于runtime机制实现的
当类A
的属性propertyA
第一次被观察时,系统就会在运行期动态地创建该类的一个派生类:NSKVONotifying_A
,在这个派生类中重写A类
中被观察属性的setter
方法以及class
,dealloc
,_isKVOA
方法。派生类在被重写的setter
方法内实现真正的通知机制。
每个类对象中都有一个isa
指针指向当前类,当类对象第一次被观察,那么系统会将isa指针指向动态生成的派生类,从而在给被监控属性赋值时执行的是派生类的setter
方法
键值观察通知依赖于NSObject
的两个方法: willChangeValueForKey:
和 didChangevlueForKey:
;在一个被观察属性发生改变之前, willChangeValueForKey:
一定会被调用,这就会记录旧的值。而当改变发生后,didChangeValueForKey:
会被调用,继而 observeValueForKey:ofObject:change:context:
也会被调用。用代码显示就是:
-(void)setName:(NSString *)newName{
[self willChangeValueForKey:@"name"]; //KVO 在调用存取方法之前总调用
[super setValue:newName forKey:@"name"]; //调用父类的存取方法
[self didChangeValueForKey:@"name"]; //KVO 在调用存取方法之后总调用
}
这里需要注意的一个小问题是:观察者观察的是属性,只有遵循 KVO 变更属性值的方式才会执行 KVO 的回调方法,例如是否执行了 setter 方法、或者是否使用了 KVC 赋值。
如果赋值没有通过 setter 方法或者 KVC,而是直接修改属性对应的成员变量,例如:仅调用_name = @"newName"
,这时是不会触发 KVO 机制,更加不会调用回调方法的。
所以使用 KVO 机制的前提是遵循 KVO 的属性设置方式来变更属性值。
上文提到的动态修改了isa
指针,isa
指针的作用:每个对象都有 isa 指针,指向该对象的类,它告诉 Runtime 系统这个对象的类是什么。所以对象注册为观察者时,isa 指针指向新子类,那么这个被观察的对象就神奇地变成新子类的对象(或实例)了。即重写class方法,是为了我们调用它的时候返回跟重写之前同样的内容。
NSLog(@"self->isa:%@",self->isa);
NSLog(@"self class:%@",[self class]);
在建立KVO监听前,打印结果为:
self->isa:Person
self class:Person
在建立KVO监听之后,打印结果为:
self->isa:NSKVONotifying_Person
self class:Person
此外派生类还重写了 dealloc 方法来释放资源,具体的笔者没有深究,网上的文章也很少有提及的。
KVO不得不关注的要点
大家可以看这篇文章,主要就是:
1、对同一个keypath进行两次removeObserver时会导致程序crash(多出现在父类也对同一个keypath监听,dealloc中也removeObserver);
2、不需要监听的时候,要及时remove,否则被监听对象释放后,再触发监听器会引起崩溃
3、父类也实现了KVO时的注意点,子类监听回调中需要做处理。
iOS下KVO使用过程中的陷阱
题外:如何手动触发KVO
近期同事问我,如何手动触发一个value的KVO?
如果上述的内容看过之后,相信就比较容易了。只需要重写待监听类的+(BOOL)automaticallyNotifiesObserversForKey:
方法,然后重写setter
方法,在赋值前后分别加上willChangeValueForKey:
和didChangeValueForKey
即可,当然监听类仍需addObserver。
总结
以上就是笔者对KVO的认识了,当然也借鉴了网上的一些博文的知识,如有错误,还请大家评论指出,大家一起学习,谢谢大家!
网友评论