------------------KVC---------------------
KVC是一種通過(guò)字符串來(lái)訪問(wèn)實(shí)例對(duì)象屬性或變量的機(jī)制,使用最多的是用來(lái)字典轉(zhuǎn)模型。利用runtime獲取對(duì)象的所有成員變量爸舒, 再根據(jù)kvc鍵值賦值羡棵,進(jìn)行字典轉(zhuǎn)模型
當(dāng)給對(duì)象發(fā)送setValue:forKey 消息時(shí)要判斷對(duì)象是否存在key所對(duì)應(yīng)的屬性,如果有壹若,直接賦值;如果沒有就調(diào)用undefinedKey(默認(rèn)崩潰,需要重寫)
setValue:forKey的調(diào)用順序
- 首先會(huì)按照setkey,_setKey順序找方法
- 如果沒有那么按 _key, _isKey皂冰,key, iskey的順序搜索成員變量名
- 如何還沒找到就調(diào)用setValue:forUndefinedKey:(默認(rèn)崩潰,需要重寫)
ValueForKey的調(diào)用順序
- 首先會(huì)按照getKey, key, isKey,_key順序找方法
- 如果沒有那么按 _key, _isKey店展,key, iskey的順序搜索成員變量名
- 如何還沒找到就調(diào)用valueForUndefinedKey
------------------KVO---------------------
KVO是觀察者模式的實(shí)現(xiàn),使用了ISA混寫技術(shù)
當(dāng)被監(jiān)聽對(duì)象某個(gè)屬性發(fā)生改變時(shí)秃流,監(jiān)聽該屬性值變化的對(duì)象可以接受到通知壁查,然后通過(guò)kvo提供的系統(tǒng)的方法響應(yīng)一些操作,有利于兩個(gè)類間的解耦
原理
KVO是基于runtime機(jī)制實(shí)現(xiàn)的剔应,某個(gè)對(duì)象被觀察時(shí),runtime會(huì)在運(yùn)行時(shí)動(dòng)態(tài)創(chuàng)建一個(gè)新的繼承被監(jiān)聽類的子類(NSKVONotifying_ 開頭)
1、然后將被監(jiān)聽類的對(duì)象的isa指針指向子類類對(duì)象峻贮;
2席怪、并重寫子類被觀察屬性的setter 方法,(這個(gè)重寫是在運(yùn)行時(shí)而不是編譯時(shí)實(shí)現(xiàn)的纤控,而且是KVO本質(zhì))挂捻;當(dāng)被監(jiān)聽對(duì)象的屬性改變時(shí),會(huì)觸發(fā)set方法船万,但這個(gè)方法被重寫了刻撒,并且在內(nèi)部調(diào)用了didChangeValueForKey方法從而觸發(fā)通知機(jī)制
KVO機(jī)制的特點(diǎn)和觸發(fā)前提
- 修改的是屬性
- 通過(guò)setter方法直接修改
- 通過(guò) KVC (內(nèi)部會(huì)調(diào)用setter方法,從而觸發(fā)KVO)
-
修改的是成員變量
KVC修改成員變量觸發(fā)KVO.png
- KVC修改內(nèi)部會(huì)觸發(fā)KVO
- 手動(dòng)觸發(fā)
KVO使用步驟
- 注冊(cè)觀察者耿导,實(shí)施監(jiān)聽
[self.p1 addObserver:self forKeyPath:@"name" options:NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld context:nil];
- 在回調(diào)方法中處理屬性發(fā)生的變化
// 這個(gè)方法時(shí)屬于 NSObject 類的声怔,任何對(duì)象都可以作為觀察者
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSString *,id> *)change context:(void *)context
{
NSLog(@"監(jiān)聽到了%@的%@屬性發(fā)生了改變", object, keyPath);
NSLog(@"%@", change);
}
- 移除觀察者
[p1 removeObserver:self forKeyPath:@"name"];
- 子類重寫被觀察屬性的setter方法的內(nèi)部實(shí)現(xiàn)
- (void)setName:(NSString *)name
{
[self willChangeValueForKey:@"age"];
[super setName:name];
[self didChangeValueForKey:@"age"];
// 這兩個(gè)方法底層會(huì)調(diào)用observer的- (void)observeValueForKeyPath: ofObject: change: context:這個(gè)方法
}
拓展
- KVO和notification(通知)的區(qū)別?
- 兩者都是一對(duì)多
- notification的優(yōu)點(diǎn)是監(jiān)聽不局限于屬性的變化舱呻,還可以對(duì)多種多樣的狀態(tài)變化進(jìn)行監(jiān)聽醋火,監(jiān)聽范圍廣
- KVO與delegate的不同?
KVO和NSNotification 都是負(fù)責(zé)發(fā)送接收通知箱吕,剩下的事情由系統(tǒng)處理芥驳,所以不用返回值; delegate 則需要代理聯(lián)系
delegate一般是一對(duì)一茬高,而這兩個(gè)可以一對(duì)多
另外需要注意的是兆旬,由于KVO這種繼承方式的注入是在運(yùn)行時(shí)而不是編譯時(shí)實(shí)現(xiàn)的,如果給定的實(shí)例沒有觀察者怎栽,那么KVO不會(huì)有任何開銷丽猬,因?yàn)榇藭r(shí)根本就沒有KVO代碼存在。但是即使沒有觀察者婚瓜,委托和NSNotification還是得工作宝鼓,這也是KVO此處零開銷觀察的優(yōu)勢(shì)
總結(jié)
對(duì)比其他的回調(diào)方式,KVO機(jī)制的運(yùn)用的實(shí)現(xiàn)巴刻,更多的由系統(tǒng)支持愚铡,相比notification、delegate等更簡(jiǎn)潔些胡陪,并且能夠提供觀察屬性的最新值以及原始值沥寥;但是相應(yīng)的在創(chuàng)建子類、重寫方法等等方面的內(nèi)存消耗是很巨大的
1柠座、異步:監(jiān)聽通知 主線程:發(fā)出通知 接收通知代碼在主線程
2邑雅、主線程:監(jiān)聽通知 異步:發(fā)出通知 接收通知代碼在異步
總結(jié):接收通知代碼 由 發(fā)出通知線程決定, KVO也一樣