KVO/KVC的底層原理和使用

KVO(Key-Value-Observe 鍵值觀察)的原理并不復(fù)雜腊满,但是涉及到isa指針钢拧、superClass指針以及runtimeOC消息分發(fā)的知識雄妥,非常容易發(fā)散最蕾,所以一直是面試熱點。

KVC鍵值編碼 是Key Value Coding 的簡稱老厌,cocoa的標(biāo)準(zhǔn)組成部分瘟则,是一種可以直接通過字符串的名字(Key)來訪問類屬性的機制,而不是通過調(diào)用Setter方法枝秤、Getter方法進行訪問醋拧。

面試

(文末回答,也請評論你遇到的面試問題淀弹,共同進步丹壕。)

如何手動實現(xiàn)KVO? 如何解除KVO?KVO優(yōu)缺點?
KVC是什么原理?能夠使用KVO監(jiān)聽嗎薇溃?
KVC賦值異常處理

原理

被添加監(jiān)聽的類Person菌赖,會在運行時動態(tài)創(chuàng)建一個該類的子類NSKVONotifying_ Person(superClass指向Person, isa指向自己的元類)痊焊。runtime會動態(tài)更改Person類的實例對象person的isa指向盏袄。當(dāng)執(zhí)行person.ageset方法時,會根據(jù)isa找到person 的類對象薄啥,找到setAge:方法(setAge:會執(zhí)行Foundation框架的一個C方法_NSSetIntValueAndNotify辕羽。_NSSetIntValueAndNotify的實現(xiàn)偽代碼如下:

{
     [self.person1 willChangeValueForKey:@"age"];
     [super setAge:10];
     [self.person1 didChangeValueForKey:@"age"];
}

didChangeValueForKey:會觸發(fā)監(jiān)聽方法 [observer observeValueForKeyPath:key ofObject:self change:change context:NULL]; 。

[super setAge:10];會執(zhí)行父類的setAge方法垄惧。

添加監(jiān)聽的實例對象結(jié)構(gòu)圖
未添加監(jiān)聽的實例對象結(jié)構(gòu)圖

用途:

主要用于監(jiān)聽屬性值的變化刁愿。可用于MVVMviewModelView的交互到逊。(請在評論區(qū)繼續(xù)ADD...)

擴展:

動態(tài)創(chuàng)建類

動態(tài)創(chuàng)建類參數(shù):父類铣口,類名,額外的內(nèi)存空間

Class objc_allocateClassPair(Class superclass, const char *name, size_t extraBytes)

如何更改isa指向和isa指針的結(jié)構(gòu)觉壶?

修改設(shè)置isa指向:

 object_setClass(id obj, Class cls)

isa和superClass指向:

  • 實例對象的isa 指針類對象脑题,類對象的isa指針指向metaClass,metaClass的isa指針指向基類NSObject.

  • 實例對象沒有superClass指針铜靶,類對象的superClass指向父類對象叔遂,一直到基類的類對象[NSObject class], NSObject的類對象指向nil。

  • metaClass對象的superClass指向父類的metaClass對象,一直到基類的metaClass對象, NSObject的metaClass對象指向類對象[NSObject class]已艰。

isa和superClass指向

面試參考答案

如何手動實現(xiàn)KVO?

1痊末、手動創(chuàng)建子類,并修改實例對象isa指向:
2哩掺、重寫set方法凿叠,+class方法
3、重寫didChangeValueForKey:

如何解除KVO?

重寫didChangeValueForKey:

KVC/KVO的優(yōu)缺點

  • KVC優(yōu)點:沒有property的變量(私有)也能通過KVC進行設(shè)置嚼吞,json或者簡化代碼(多級屬性)或者json轉(zhuǎn)model 簡化代碼
  • KVC缺點:如果key只寫錯盒件,編寫的時候不會報錯,但是運行的時候會報錯

KVO優(yōu)點

  • 能夠提供一種簡單的方法實現(xiàn)兩個對象的同步誊薄;
  • 能夠?qū)?nèi)部對象的狀態(tài)改變作出響應(yīng)履恩,而且不需要改變內(nèi)部對象的實現(xiàn);
  • 能夠提供被觀察者屬性的最新值和之前的值呢蔫;
  • 使用key Path來觀察屬性切心,因此可以觀察嵌套對象;
  • 完成了對觀察對象的抽象片吊,因為不需要額外的代碼來允許觀察者被觀察绽昏。

KVO缺點

  • KVO只能檢測類中的屬性,并且屬性名都是通過NSString來查找俏脊,編譯器不會補全(編譯時不會出現(xiàn)警告)全谤,容易寫錯;
  • 對屬性重構(gòu)爷贫,將導(dǎo)致觀察代碼不可用认然;
  • 復(fù)雜的 “if” 語句要求對象正在觀察多個值,是因為所有的觀察代碼通過一個方法來指向漫萄;

KVC能夠使用KVO監(jiān)聽嗎

KVC的API如下所示:

- (void)setValue:(id)value forKeyPath:(NSString *)keyPath;
- (void)setValue:(id)value forKey:(NSString *)key;
- (id)valueForKeyPath:(NSString *)keyPath;
- (id)valueForKey:(NSString *)key; 

KVC訪問變量的流程如下圖所示:


setValueforkey的執(zhí)行流程

setValueforkey首先調(diào)用的是setKey方法卷员,OC屬性聲明后或自動生成set 、 get 方法和_key的局部變量腾务,所以默認(rèn)是可以被KVO監(jiān)聽到的毕骡。
但是如果屬性被readOnly修飾就不會自動生成set方法, 但是如果用KVC的話仍然可以修改被readOnly修飾的值岩瘦。而且能夠出發(fā)KVO監(jiān)聽未巫,證明了下面的流程圖:尋找_key的局部變量直接修改,并且主動調(diào)用willChangeValueForKey 和didChangeValueForKey启昧, 觸發(fā)KVO監(jiān)聽叙凡。(思考一下這是readOnly的漏洞嗎?怎么解決呢密末?評論區(qū)見)

getValueforkey的執(zhí)行流程

KVC賦值異常處理

- (void)setNilValueForKey:(NSString *)key
{
    NSLog(@"這里處理當(dāng)賦值為nil時握爷,出現(xiàn)異常");
}
- (void)setValue:(id)value forUndefinedKey:(NSString *)key
{
    NSLog(@"key沒有定義的時候宰啦,可以在這里處理");
}

如有錯誤或者新的見解歡迎在評論區(qū)約談...

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市饼拍,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌田炭,老刑警劉巖师抄,帶你破解...
    沈念sama閱讀 221,635評論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異教硫,居然都是意外死亡叨吮,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,543評論 3 399
  • 文/潘曉璐 我一進店門瞬矩,熙熙樓的掌柜王于貴愁眉苦臉地迎上來茶鉴,“玉大人,你說我怎么就攤上這事景用『#” “怎么了?”我有些...
    開封第一講書人閱讀 168,083評論 0 360
  • 文/不壞的土叔 我叫張陵伞插,是天一觀的道長割粮。 經(jīng)常有香客問我,道長媚污,這世上最難降的妖魔是什么舀瓢? 我笑而不...
    開封第一講書人閱讀 59,640評論 1 296
  • 正文 為了忘掉前任,我火速辦了婚禮耗美,結(jié)果婚禮上京髓,老公的妹妹穿的比我還像新娘。我一直安慰自己商架,他們只是感情好堰怨,可當(dāng)我...
    茶點故事閱讀 68,640評論 6 397
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著甸私,像睡著了一般诚些。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上皇型,一...
    開封第一講書人閱讀 52,262評論 1 308
  • 那天诬烹,我揣著相機與錄音,去河邊找鬼弃鸦。 笑死绞吁,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的唬格。 我是一名探鬼主播家破,決...
    沈念sama閱讀 40,833評論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼颜说,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了汰聋?” 一聲冷哼從身側(cè)響起门粪,我...
    開封第一講書人閱讀 39,736評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎烹困,沒想到半個月后玄妈,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,280評論 1 319
  • 正文 獨居荒郊野嶺守林人離奇死亡髓梅,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,369評論 3 340
  • 正文 我和宋清朗相戀三年拟蜻,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片枯饿。...
    茶點故事閱讀 40,503評論 1 352
  • 序言:一個原本活蹦亂跳的男人離奇死亡酝锅,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出奢方,到底是詐尸還是另有隱情搔扁,我是刑警寧澤,帶...
    沈念sama閱讀 36,185評論 5 350
  • 正文 年R本政府宣布袱巨,位于F島的核電站阁谆,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏愉老。R本人自食惡果不足惜场绿,卻給世界環(huán)境...
    茶點故事閱讀 41,870評論 3 333
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望嫉入。 院中可真熱鬧焰盗,春花似錦、人聲如沸咒林。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,340評論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽垫竞。三九已至澎粟,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間欢瞪,已是汗流浹背活烙。 一陣腳步聲響...
    開封第一講書人閱讀 33,460評論 1 272
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留遣鼓,地道東北人啸盏。 一個月前我還...
    沈念sama閱讀 48,909評論 3 376
  • 正文 我出身青樓,卻偏偏與公主長得像骑祟,于是被迫代替她去往敵國和親回懦。 傳聞我的和親對象是個殘疾皇子气笙,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 45,512評論 2 359

推薦閱讀更多精彩內(nèi)容