KVO

作者: Small_Potato | 来源:发表于2019-03-13 16:12 被阅读0次

    一:简单的说下KVO

    当观察一个对象时,一个新的类会动态被创建。这个类继承自该对象的原本的类,并重写了被观察属性的 setter 方法。自然,重写的 setter 方法会负责在调用原 setter 方法之前和之后,通知所有观察对象值的更改(willChangeValueForKey:和 didChangevlueForKey:)。最后把这个对象的 isa 指针 ( isa 指针告诉 Runtime 系统这个对象的类是什么 ) 指向这个新创建的子类,对象就神奇的变成了新创建的子类的实例。

    具体可以看下面:
    1:刚开始创建的时候_p对象的isa指针指向了Person这个类;
    2:添加KVO后我们可以看到,_p的isa指针指向了NSKVONotifying_Person这个类,这个类就是系统帮我们创建的类;


    1
    2

    二:简单的说下如何自己实现KVO

    通过上面,我们知道了KVO的过程其实就是系统帮我们创建了个继承自当前类的子类,然后重写了set方法,在里面通知值的改变(拓展:子类并没有实现父类的方法,我们调用子类的方法的时候,现在当前类的方法列表中找不到,然后实际是去父类的方法列表里面寻找了,所以我们说是重写).,按照这个思路简单的写下;

    1.首先创建个Category,我们ZXKVO;
    2.我们在Category中添加一个方法用于实现自己的KVO;
     2.1首先在我们的方法中,创建一个类,然后重写这个类的set方法,并把当前类的isa指向我们新创建的类,如图3;
     2.2重写set函数,在函数中 我们需要调用父类的set方法,然后获取到绑定的观察者,把事件通知给观察者就行了,如图4;


    3
    4

    备注:
    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方法(有兴趣的可以自己去看看).

    image.png

    以后有时间了在补充下用RAC(函数式响应式)实现KVO,例如AFN就是有这样的实现,直接用block实现对使用者来说会非常的可观简单.

    就到这吧,比较忙,不多BB,最后附上代码链接

    相关文章

      网友评论

          本文标题:KVO

          本文链接:https://www.haomeiwen.com/subject/bpawyftx.html