我们平常使用KVO分为以下三个步骤:1.创建 2:订阅 3:销毁
当KVO创建完成后,如上图所示,P的isa指针发生了改变,指向了NSKVONotifying_Person,当观察的值发生改变时就会马上调用订阅方法,那么在这内部到底发生了什么?接下来我们一探究竟
由于Objective-C源码不是开源的,我们用GNUStep代替
方框部分声明了一些变量,我们首先来看箭头指向的 r = replacementForClass([selfclass]);
首先看是否已经有传入的c这个类所生成的GSKVORelacement的对象,如果没有则初始化一个,并把它插入到Map中,接下来我们看下GSKVORelacement初始化做了什么
前面做了一些变量的声明和一些判断,重点看到红框的部分,是不是有些熟悉,那我们来简单的分析一下
1.superName = NSStringFromClass(original);调用Runtime方法获取到传入类的名称
2.name = [@"GSKVO"stringByAppendingString: superName];对传入类的名称做了一个拼接,如图2所示NSKVONotifying_Person。 看到这里我们似乎对图2中为什么出现NSKVONotifying_Person有一点明白了
注:这里由于使用的是GNUStep的源码所以和Objective-C真实情况有一点偏差
3.template = GSObjCMakeClass(name, superName,nil);这里做了什么呢
看到箭头部分我们明白了,初始化了一个新的继承于传入类的类对
4.GSObjCAddClasses([NSArray arrayWithObject: template]);
把第三步初始化的类对进行注册
info = (GSKVOInfo*)[selfobservationInfo]; 获取观察需要的信息,observer,KeyPath,options,context等信息
而后info 又去调用了 - (void)addObserver: (NSObject*)anObserver forKeyPath: (NSString*)aPath options: (NSKeyValueObservingOptions)options context: (void*)aContext; 方法
[pathInfo->change setObject: [NSNumber numberWithInt:1] forKey: NSKeyValueChangeKindKey];这里调用用了这个方法
在图1中我们看到self.p.name = @"moul";通过set方法来修改值的
在set方法中调用了willChangeValueForKey方法
为了能看到整个方法的代码,修改了下源代码中的格式
看下[pathInfo notifyForKey: aKey ofInstance: [info instance] prior:YES];
当看到方框中的代码时一切都明白了,在这里内部调用了- (void)observeValueForKeyPath:(NSString*)keyPathofObject:(id)objectchange:(NSDictionary *)changecontext:(void*)context这个方法把观察的值传了出来
那么我们的KVO探索也就清楚了
下面来总结下KVO的实现过程
1.生成一个子类 NSKVONotifying_XX
2.改变父类的isa指针指向子类
3.在子类中添加setter方法
4.在子类中重写属性的setter方法
5.通知观察者调用方法后传出
以下是自定义KVO的代码