KVO(Key-Value-Observe
鍵值觀察)的原理并不復(fù)雜腊满,但是涉及到isa指針
钢拧、superClass指針
以及runtime
和 OC消息分發(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.age
的set
方法時,會根據(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)聽屬性值的變化刁愿。可用于MVVM
中 viewModel
和View
的交互到逊。(請在評論區(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]已艰。
面試參考答案
如何手動實現(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首先調(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ū)見)
KVC賦值異常處理
- (void)setNilValueForKey:(NSString *)key
{
NSLog(@"這里處理當(dāng)賦值為nil時握爷,出現(xiàn)異常");
}
- (void)setValue:(id)value forUndefinedKey:(NSString *)key
{
NSLog(@"key沒有定義的時候宰啦,可以在這里處理");
}
如有錯誤或者新的見解歡迎在評論區(qū)約談...