KVO底層實現(xiàn)原理

KVO的本質(zhì)

key-value observer,鍵值監(jiān)聽,對某一個實例對象進行該實例對象某一個屬性值的監(jiān)聽addObserver:forKeyPath:options:content:對象添加監(jiān)聽的方法

參數(shù)

observer,需要誰(實例對象)來監(jiān)聽

keyPath,需要監(jiān)聽的屬性名稱 不是所有的屬性都可以監(jiān)聽

options,監(jiān)聽該屬性的方式 新值 / 舊值

content,攜帶值,一般為nil

KVO 不會對self進行強引用,如果觀察者的生命周期結(jié)束了,self被釋放,所以添加的監(jiān)聽需要被移除。

iOS用什么方式實現(xiàn)對一個對象的KVO?(KVO的本質(zhì)是什么?)

利用RuntimeAPI動態(tài)生成一個子類NSKVONotifying,并且讓instance對象的isa指向這個全新的子類,當(dāng)修改instance對象的屬性時,會調(diào)用Foundation的_NSSetXXXValueAndNotify函數(shù)willChangeValueForKey:

父類原來的setter

didChangeValueForKey:

內(nèi)部會觸發(fā)監(jiān)聽器(Oberser)的監(jiān)聽方法(observeValueForKeyPath:ofObject:change:context:)

如何手動觸發(fā)KVO?

手動調(diào)用willChangeValueForKey:和didChangeValueForKey:

KVO

kvo

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

實例對象的isa指針指向類對象,因為類對象在內(nèi)存中存儲的信息有isa指針,superclass,屬性,對象方法,協(xié)議,成員變量等,因此可以在類對象中找到對象方法,進行調(diào)用。
使用了KVO監(jiān)聽的對象

當(dāng)一個實例對象調(diào)用添加監(jiān)聽屬性的方法時,實際上是runtime動態(tài)創(chuàng)建了一個名為“NSKVONotifying_Person類名”的類,該類會繼承與person類,并且讓該實例對象的isa指針重新指向了這個動態(tài)創(chuàng)建的類的類對象。因此在該實例對象屬性修改值的時候,實際上在內(nèi)存中該實例對象通過isa指針找到了動態(tài)創(chuàng)建的這個子類的類對象,這個動態(tài)的類對象中重寫了屬性的set方法,只是這個動態(tài)類的set方法實現(xiàn)并不是直接修改該屬性的值,而是調(diào)用了一個名叫 “_NSSetIntValueAndNotify的C方法。

_NSSetIntValueAndNotify方法的內(nèi)部實現(xiàn),相當(dāng)于在調(diào)用原來的set方法實現(xiàn)之前,加入 willChangeValueForKey,在其方法后,加入didChangeValueForKey,以此來記錄屬性的新值和舊值。然后在重寫didChangeValueForKey方法中,通過調(diào)用 observer的observerValueForKeyPath:ofObject:change:content方法通知監(jiān)聽者。

NSSetxxxValueAndNotify內(nèi)部實現(xiàn)

直接修改成員變量會觸發(fā)KVO嗎?

不會 因為沒有調(diào)用set方法,KVO的本質(zhì)就是重寫set方法

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容