KVO是基于runtime机制实现的。当某个类的属性对象第一次被观察时,系统就会在运行期动态地创立该类的一个派生类,在这个派生类中重写基类中任何被观察属性的setter方法。派生类在被重写的setter方法内实现真正的通知机制。
以Person类为例,那么生成的派生类名为NSKVONotifying_Person。
当某个属性第一次被观察,那么系统会偷偷将其isa指针从原来的类,改为指向动态生成的派生类,从而使得属性被赋值时执行的是派生类的setter方法。
在派生类重写的settter方法中, 通过NSObject的 willchangeValueforKey、didchangeValueForKey等方法通知外部观察者,继而使得观察者的observeValueForKey:ofObject:change:context:方法被调用。
//NSKVONotifying_Person的setter方法
- (void) setValue: (id) obj{
[self willChangeValueForKey:@"keyPath"];
[super setValue: obj];
[self didChangeValueForKey:@"keyPath"];
}
问题: 通过给成员变量直接赋值,而不是通过属性修改成员变量在变量值变化时收到通知?
答案是否定的。因为kvo是重写的setter方法,成员变量赋值只是修改指针的引用,并不会调用setter方法。