KVO的本質(zhì)
key-value observer,鍵值監(jiān)聽,對某一個(gè)實(shí)例對象進(jìn)行該實(shí)例對象某一個(gè)屬性值的監(jiān)聽addObserver:forKeyPath:options:content:對象添加監(jiān)聽的方法
參數(shù)
observer,需要誰(實(shí)例對象)來監(jiān)聽
keyPath,需要監(jiān)聽的屬性名稱 不是所有的屬性都可以監(jiān)聽
options,監(jiān)聽該屬性的方式 新值 / 舊值
content,攜帶值,一般為nil
KVO 不會(huì)對self進(jìn)行強(qiáng)引用,如果觀察者的生命周期結(jié)束了,self被釋放,所以添加的監(jiān)聽需要被移除。
iOS用什么方式實(shí)現(xiàn)對一個(gè)對象的KVO?(KVO的本質(zhì)是什么?)
利用RuntimeAPI動(dòng)態(tài)生成一個(gè)子類NSKVONotifying,并且讓instance對象的isa指向這個(gè)全新的子類,當(dāng)修改instance對象的屬性時(shí),會(huì)調(diào)用Foundation的_NSSetXXXValueAndNotify函數(shù)willChangeValueForKey:
父類原來的setter
didChangeValueForKey:
內(nèi)部會(huì)觸發(fā)監(jiān)聽器(Oberser)的監(jiān)聽方法(observeValueForKeyPath:ofObject:change:context:)
如何手動(dòng)觸發(fā)KVO?
手動(dòng)調(diào)用willChangeValueForKey:和didChangeValueForKey:
KVO

未使用KVO監(jiān)聽的對象

實(shí)例對象的isa指針指向類對象,因?yàn)轭悓ο笤趦?nèi)存中存儲(chǔ)的信息有isa指針,superclass,屬性,對象方法,協(xié)議,成員變量等,因此可以在類對象中找到對象方法,進(jìn)行調(diào)用。
使用了KVO監(jiān)聽的對象

當(dāng)一個(gè)實(shí)例對象調(diào)用添加監(jiān)聽屬性的方法時(shí),實(shí)際上是runtime動(dòng)態(tài)創(chuàng)建了一個(gè)名為“NSKVONotifying_Person類名”的類,該類會(huì)繼承與person類,并且讓該實(shí)例對象的isa指針重新指向了這個(gè)動(dòng)態(tài)創(chuàng)建的類的類對象。因此在該實(shí)例對象屬性修改值的時(shí)候,實(shí)際上在內(nèi)存中該實(shí)例對象通過isa指針找到了動(dòng)態(tài)創(chuàng)建的這個(gè)子類的類對象,這個(gè)動(dòng)態(tài)的類對象中重寫了屬性的set方法,只是這個(gè)動(dòng)態(tài)類的set方法實(shí)現(xiàn)并不是直接修改該屬性的值,而是調(diào)用了一個(gè)名叫 “_NSSetIntValueAndNotify的C方法。
_NSSetIntValueAndNotify方法的內(nèi)部實(shí)現(xiàn),相當(dāng)于在調(diào)用原來的set方法實(shí)現(xiàn)之前,加入 willChangeValueForKey,在其方法后,加入didChangeValueForKey,以此來記錄屬性的新值和舊值。然后在重寫didChangeValueForKey方法中,通過調(diào)用 observer的observerValueForKeyPath:ofObject:change:content方法通知監(jiān)聽者。

直接修改成員變量會(huì)觸發(fā)KVO嗎?
不會(huì) 因?yàn)闆]有調(diào)用set方法,KVO的本質(zhì)就是重寫set方法