KVO
KVO的全稱是Key-Value Observing,俗稱“鍵值監(jiān)聽(tīng)”匿乃,可以用于監(jiān)聽(tīng)某個(gè)對(duì)象屬性值的改變
假如我們有一個(gè)Person類,并且使用KVO監(jiān)聽(tīng)age.
這時(shí)候我們發(fā)現(xiàn)如果我們使用
person->_age = 1;
會(huì)發(fā)現(xiàn)kvo監(jiān)聽(tīng)失效
但是使用
person.age = 1;
kvo是可以實(shí)現(xiàn)監(jiān)聽(tīng)的
根據(jù)兩次賦值我們來(lái)深究一下kvo到底是如何實(shí)現(xiàn)的
前置:我們生成兩個(gè)person的對(duì)象為person1和person2 并且監(jiān)聽(tīng)person1的age屬性
然后我們使用runtime來(lái)的object_getClass來(lái)顯示出倆個(gè)person對(duì)象的類名 使用methodForSelector顯示出setAge的方法地址 并且打印我們得到一下結(jié)果:
根據(jù)打印結(jié)果我們得知在person1未被kvo監(jiān)聽(tīng)的情況下 兩個(gè)person對(duì)象的類都是Person且setage:都是一個(gè)相同的地址.但是在person1被kvo監(jiān)聽(tīng)后其類對(duì)象變成啦NSKVONotifying_Person 且setage也發(fā)生啦改變
由此我們得知在我們使用kvo監(jiān)聽(tīng)的時(shí)候系統(tǒng)內(nèi)部動(dòng)態(tài)生成啦一個(gè)Person的新子類且重新的Person類的set方法(因?yàn)槲覀冊(cè)诘谝淮尾捎米兞抠x值時(shí)候kvo并未監(jiān)聽(tīng)成功而采用set賦值的時(shí)候kvo監(jiān)聽(tīng)成功)
那么kvo在動(dòng)態(tài)生成的Person子類到底從寫(xiě)啦那些方法并且重寫(xiě)啦些什么東西呢?
我們使用以下方法來(lái)獲得子類的相關(guān)方法名
得出結(jié)果重寫(xiě)啦?Person類的set:, class, dealloc, _isKVOA
那么他在重寫(xiě)方面的方法內(nèi)部添加什么呢?
那么我們使用lldb imp得到一下結(jié)果
由此我們得出結(jié)論在kvo生成NSKVONotifying_Person內(nèi)部吧setage方法替換成了_NSSetIntValueAndNotify來(lái)實(shí)現(xiàn)監(jiān)聽(tīng)但是這個(gè)方法的是實(shí)現(xiàn)是在Foundation內(nèi)部我們無(wú)法得知具體實(shí)現(xiàn)
根據(jù)網(wǎng)上一些大神的翻譯解包我們大致能了解到其實(shí)內(nèi)部大致是
[self willChangeValueForKey:@"age"];
[super setAge:age];
[self didChangeValueForKey:@"age"];
那么我們?cè)赑erson內(nèi)部重寫(xiě)一下這兩個(gè)方法
那么其他方法是重寫(xiě)啦寫(xiě)什么呢??
我們調(diào)用person1.class方法發(fā)現(xiàn)打印仍然是Person類得出結(jié)果系統(tǒng)并不是想讓你知道他動(dòng)態(tài)生成啦一個(gè)子類所以重寫(xiě)啦class方法告訴你還是Person類
其他兩個(gè)方法我并沒(méi)有去驗(yàn)證所以不發(fā)表意見(jiàn)
一下是用一張圖來(lái)總結(jié)一下
KVC
對(duì)于kvc我們簡(jiǎn)單用兩張圖可以概況 但是我們先提出一個(gè)疑問(wèn) kvc賦值kvc監(jiān)聽(tīng)會(huì)觸發(fā)嗎?
答案是會(huì)的 并且就算你是變量賦值也會(huì)觸發(fā)有興趣的小伙伴可以自己去嘗試一下大概實(shí)現(xiàn)猜想是kvc在賦值以后會(huì)主動(dòng)調(diào)用willChangeValueForKey:和didChangeValueForKey:
setValue:forKey:
valueForKey