KVO是什么
KVO 是 OC 对 观察者模式的实现。
KVO提供一种机制,指定一个被观察对象A,当A的某个属性(如name)发生改变,对象就会收到通知,并作出相应处理
在MVC设计架构下,很适合实现model模型和view视图之间的通讯。
KVO实现原理
当观察者对象A时,KVO机制动态创建了一个对象A当前类的子类,并未这个新的子类重写了被观察属性keyPath的setter方法,setter方法随后负责通知观察对象属性的改变情况。
Apple使用了isa混写(isa-swizzling)来实现KVO。当观察对象A时,KVO机制动态创建一个名为NSKVONotifying_A的新类,该类继承自对象A的本类,且KVO为NSKVONotifying_A重写观察属性的setter方法,setter方法会负责在调用原setter方法之前和之后,通知所有观察对象属性职的更改情况。
解析NSKVONotifying_A类
在这个过程,被观察对象的isa指针从指向原来的A类,被KVO机制修改为指向 系统新创建的NSKVONotifying_A类,来实现当前类属性值改变的监听。
所以当我们从应用层面上看来,完全没有意识到有新的类出现,如果创建一个名为“NSKVONotifying_A”的类,系统运行到注册KVO的那段代码程序就会崩溃,因为系统在注册监听的时候动态创建了名为“NSKVONotifying_A”的中间类,并指向这个中间类。
isa指针的作用,每个对象都有isa指针,指向该对象的类,它告诉runtime系统这个对象的类是什么。所以对象注册为观察者是,isa指针指向新子类,那么这个被观察的对象就神奇的变成了新子类的对象(或实例)了。因我而在该对象上对setter的调用会调用已重写的setter,从而激活键值通知机制。
子类setter方法剖析
KVO的键值观察通知依赖于NSObject的两个方法:willChangeValueForKey:和didChangeValueForKey: 在存取数值的前后分别调用2个方法。
被观察属性发生改变前,willChangeForKey被调用,通知系统该keyPath的属性值即将改变。
当改变发生后,didChangeValueForKey:被调用,通知系统改keyPath值已经变更。
之后,observeValueForKey:ofObject:change:context:也会被调用。
且重写观察属性的setter方法这种继承方式的注入是在运行时而不是编译时实现的。
KVO为子类的观察者属性重写调用存取放的工作原理在代码中相当于:
-(void)setName:(NSString *)newName {
[self willChangeValueForKey:@"name"];
[super setValue:newName forKey:@"name"];
[self didChangeValueForKey:@"name"];
}