iOS Developer

KVO的底层实现原理

2017-08-14  本文已影响93人  charlotte2018

引用孙源的话

http://blog.sunnyxx.com/2014/03/09/objc_kvo_secret/

1. 当一个object有观察者时,动态创建这个object的类的子类

2. 对于每个被观察的property,重写其set方法.

3.在重写的set方法中调用- willChangeValueForKey:和- didChangeValueForKey:通知观察者

4. 当一个property没有观察者时,删除重写的方法

5. 当没有observer观察任何一个property时,删除动态创建的子类


当某个类的属性对象第一次被观察时,系统就会在运行期动态地创建该类的一个派生类,咱们的派生类是NSKVONotifying_Person。每个类对象中都有一个isa指针指向当前类,当一个类对象的第一次被观察,那么系统会偷偷将isa指针指向动态生成的派生类,从而在给被监控属性赋值时执行的是派生类的setter方法,为什么通过p.class获取到的是Person呢?因为苹果重写了NSKVONotifying_Person这个类的class方法。返回的是父类的类名,也就是Person。让我们不知道他的内部实现。苹果还想假装一把。😄

73CC84A1-D6B7-4441-BDD7-45FC839A01CF.png

创建完了这个NSKVONotifying_Person之后,在这个派生类中重写基类中任何被观察属性的setter 方法。派生类在被重写的setter方法内实现真正的通知机制。键值观察通知依赖于NSObject 的两个方法: willChangeValueForKey: 和 didChangevlueForKey:;在一个被观察属性发生改变之前, willChangeValueForKey:一定会被调用,这就 会记录旧的值。而当改变发生后,didChangeValueForKey:会被调用,继而 observeValueForKey:ofObject:change:context: 也会被调用。

我们复写了Person 的两个方法


AC8ECC0E-06DB-4957-AE9E-3402FCB2244F.png

当我们点击屏幕的时候给name赋值

D0926E99-1288-4DAF-A83B-3958F4BC4C0A.png

看控制台的输出吧,willChangeValueForKey和didChangeValueForKey都被调用了。同时也触发了监听的方法

60D96150-B708-43F1-A71C-CB8EDAB79703.png

willChangeValueForKey和didChangeValueForKey触发了监听方法的调用

2EC85EA8-F6B9-4362-8803-C8425B678858.png

自定义一个KVO,加深对kvo的理解

参考http://tech.glowing.com/cn/implement-kvo/文章。
先熟悉下Runtime吧http://www.jianshu.com/p/f900de4a1495

头文件

74F29C07-22AA-48D1-AC81-8F9B41777B7D.png

实现

CE9A94B2-F830-4721-83BD-726D9DF45763.png

移除观察者


E646A59A-79DB-4594-9DE5-AF3D82FA7B47.png

获取KVO类

73269CE8-363F-4DC3-A6AD-6D20F32E7202.png

获取get和set方法名

319C346F-4032-4883-B85E-419C8F2FF3F4.png

KVO类重写set方法

42F734BB-9E93-4C83-9F46-BDED5E3BC8EF.png

最后说说YBObserverInfo,它保存着监听者,监听的属性 和 回调的block。当有多个监听者监听同一个属性的时候,会把所有的监听者放到一个数组里统一管理。

125A5422-FD75-45D0-A1B9-CE33F589FD11.png

用法

当我点击屏幕的时候,block会回调。但是block回调的是在多线程,如果要刷新UI,要切到主线程

CB871044-FC6D-4DA5-B318-E4350C9C529E.png

demo地址

https://github.com/yinbowang/KVODEMO

上一篇 下一篇

猜你喜欢

热点阅读