當(dāng)觀察某對象 A 時,KVO 機制動態(tài)創(chuàng)建一個對象A當(dāng)前類的子類,并為這個新的子類重寫了被觀察屬性 keyPath 的 setter 方法。setter 方法隨后負(fù)責(zé)通知觀察對象屬性的改變狀況。
深入剖析:
Apple 使用了 isa 混寫(isa-swizzling)來實現(xiàn) KVO 。當(dāng)觀察對象A時,KVO機制動態(tài)創(chuàng)建一個新的名為:NSKVONotifying_A 的新類,該類繼承自對象A的本類,且 KVO 為 NSKVONotifying_A 重寫觀察屬性的 setter 方法,setter 方法會負(fù)責(zé)在調(diào)用原 setter 方法之前和之后,通知所有觀察對象屬性值的更改情況。
(備注: isa 混寫(isa-swizzling)isa:is a kind of ; swizzling:混合,攪合;)
①NSKVONotifying_A 類剖析:在這個過程,被觀察對象的 isa 指針從指向原來的 A 類,被 KVO 機制修改為指向系統(tǒng)新創(chuàng)建的子類 NSKVONotifying_A 類,來實現(xiàn)當(dāng)前類屬性值改變的監(jiān)聽;
所以當(dāng)我們從應(yīng)用層面上看來,完全沒有意識到有新的類出現(xiàn),這是系統(tǒng)“隱瞞”了對 KVO 的底層實現(xiàn)過程,讓我們誤以為還是原來的類。但是此時如果我們創(chuàng)建一個新的名為“NSKVONotifying_A”的類,就會發(fā)現(xiàn)系統(tǒng)運行到注冊 KVO 的那段代碼時程序就崩潰,因為系統(tǒng)在注冊監(jiān)聽的時候動態(tài)創(chuàng)建了名為 NSKVONotifying_A 的中間類,并指向這個中間類了。
(isa 指針的作用:每個對象都有 isa 指針,指向該對象的類,它告訴 Runtime 系統(tǒng)這個對象的類是什么。所以對象注冊為觀察者時,isa 指針指向新子類,那么這個被觀察的對象就神奇地變成新子類的對象(或?qū)嵗┝恕?/strong>) 因而在該對象上對 setter 的調(diào)用就會調(diào)用已重寫的 setter,從而激活鍵值通知機制。
—>我猜,這也是 KVO 回調(diào)機制,為什么都俗稱KVO技術(shù)為黑魔法的原因之一吧:內(nèi)部神秘、外觀簡潔。
②子類setter方法剖析:KVO 的鍵值觀察通知依賴于 NSObject 的兩個方法:willChangeValueForKey:和 didChangevlueForKey:,在存取數(shù)值的前后分別調(diào)用 2 個方法:
被觀察屬性發(fā)生改變之前,willChangeValueForKey:被調(diào)用,通知系統(tǒng)該 keyPath 的屬性值即將變更;當(dāng)改變發(fā)生后, didChangeValueForKey: 被調(diào)用,通知系統(tǒng)該 keyPath 的屬性值已經(jīng)變更;之后, observeValueForKey:ofObject:change:context: 也會被調(diào)用。且重寫觀察屬性的 setter 方法這種繼承方式的注入是在運行時而不是編譯時實現(xiàn)的。
KVO 為子類的觀察者屬性重寫調(diào)用存取方法的工作原理在代碼中相當(dāng)于:
-(void)setName:(NSString *)newName{
[self willChangeValueForKey:@"name"]; //KVO 在調(diào)用存取方法之前總調(diào)用
[super setValue:newName forKey:@"name"]; //調(diào)用父類的存取方法
[self didChangeValueForKey:@"name"]; //KVO 在調(diào)用存取方法之后總調(diào)用
}
三、特點:
觀察者觀察的是屬性,只有遵循 KVO 變更屬性值的方式才會執(zhí)行 KVO 的回調(diào)方法,例如是否執(zhí)行了 setter 方法、或者是否使用了 KVC 賦值。
如果賦值沒有通過 setter 方法或者 KVC,而是直接修改屬性對應(yīng)的成員變量,例如:僅調(diào)用 _name = @"newName",這時是不會觸發(fā) KVO 機制,更加不會調(diào)用回調(diào)方法的。
所以使用 KVO 機制的前提是遵循 KVO 的屬性設(shè)置方式來變更屬性值。
四、步驟
- 1.注冊觀察者,實施監(jiān)聽;
- 2.在回調(diào)方法中處理屬性發(fā)生的變化;
- 3.移除觀察者
五.實現(xiàn)方法(蘋果 API 文檔中的方法):
A.注冊觀察者:
//第一個參數(shù) observer:觀察者 (這里觀察self.myKVO對象的屬性變化)
//第二個參數(shù) keyPath: 被觀察的屬性名稱(這里觀察 self.myKVO 中 num 屬性值的改變)
//第三個參數(shù) options: 觀察屬性的新值、舊值等的一些配置(枚舉值,可以根據(jù)需要設(shè)置,例如這里可以使用兩項)
//第四個參數(shù) context: 上下文,可以為 KVO 的回調(diào)方法傳值(例如設(shè)定為一個放置數(shù)據(jù)的字典)
[self.myKVO addObserver:self
forKeyPath:@"num"
options:NSKeyValueObservingOptionOld|NSKeyValueObservingOptionNew
context:nil];
B. 屬性(keyPath)的值發(fā)生變化時,收到通知,調(diào)用以下方法:
//keyPath:屬性名稱
//object:被觀察的對象
//change:變化前后的值都存儲在 change 字典中
//context:注冊觀察者時,context 傳過來的值
-(void)observeValueForKeyPath:(NSString *)keyPath
ofObject:(id)object
change:(NSDictionary<NSString *,id> *)change
context:(void *)context
{
}