一:简单的说下KVO
当观察一个对象时,一个新的类会动态被创建。这个类继承自该对象的原本的类,并重写了被观察属性的 setter 方法。自然,重写的 setter 方法会负责在调用原 setter 方法之前和之后,通知所有观察对象值的更改(willChangeValueForKey:和 didChangevlueForKey:)。最后把这个对象的 isa 指针 ( isa 指针告诉 Runtime 系统这个对象的类是什么 ) 指向这个新创建的子类,对象就神奇的变成了新创建的子类的实例。
具体可以看下面:
1:刚开始创建的时候_p对象的isa指针指向了Person这个类;
2:添加KVO后我们可以看到,_p的isa指针指向了NSKVONotifying_Person这个类,这个类就是系统帮我们创建的类;


二:简单的说下如何自己实现KVO
通过上面,我们知道了KVO的过程其实就是系统帮我们创建了个继承自当前类的子类,然后重写了set方法,在里面通知值的改变(拓展:子类并没有实现父类的方法,我们调用子类的方法的时候,现在当前类的方法列表中找不到,然后实际是去父类的方法列表里面寻找了,所以我们说是重写).,按照这个思路简单的写下;
1.首先创建个Category,我们ZXKVO;
2.我们在Category中添加一个方法用于实现自己的KVO;
2.1首先在我们的方法中,创建一个类,然后重写这个类的set方法,并把当前类的isa指向我们新创建的类,如图3;
2.2重写set函数,在函数中 我们需要调用父类的set方法,然后获取到绑定的观察者,把事件通知给观察者就行了,如图4;
3

备注:
1.在set函数中,我们把isa指针指向了父类后,一定要记得指回来;
2.class_addMethod中的"v:@:",具体的可以按住cmd+shift+0查看,这个具体的意思就是void,两个参数,一个SEL. 我们知道iOS中所有的方法调用背后都是变成了objc_msgSend(),点进去可以看到这里面有两个最基本的参数,id _Nullable self, SEL _Nonnull op,所以在set函数中,除了改变的值,还有这两个基本参数;
3.我们可以重写Person的setName方法,这个不影响,因为我们调用了super的set方法;
补充:KVO对容器的集合
我们知道KVO是通过重写set方法实现通知改变的,但是在容器中增删改查并不能调起set方法,这个时候用下面这个方法,mutableArrayValueForKey这个方法的作用也是一样的,创建子类,重新指向isa,重写set方法(有兴趣的可以自己去看看).

以后有时间了在补充下用RAC(函数式响应式)实现KVO,例如AFN就是有这样的实现,直接用block实现对使用者来说会非常的可观简单.
就到这吧,比较忙,不多BB,最后附上代码链接
网友评论