引用孫源的話
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.lxweimin.com/p/f900de4a1495
頭文件
74F29C07-22AA-48D1-AC81-8F9B41777B7D.png
實現
- (void)yb_addObserver:(NSObject *)observer forKey:(NSString *)key withBlock:(YBObserverBlock) block
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