KVO 鍵值觀察原理淺析

相信大家在日常開(kāi)發(fā)過(guò)程中都有使用過(guò)以下方法:
- (void)addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(nullable void *)context;

在我們監(jiān)聽(tīng)了某一個(gè)對(duì)象的某個(gè)屬性之后又發(fā)生了什么事情呢?


在OC中隘梨,每一個(gè)對(duì)象都是類的實(shí)例,每一個(gè)對(duì)象都有一個(gè)名為 isa 的指針,指向創(chuàng)建該對(duì)象的類澈蚌。 詳見(jiàn)Objective-C對(duì)象模型及應(yīng)用

我們創(chuàng)建兩個(gè)TestObject類的實(shí)例對(duì)象

_obj = [[TestObject alloc]init];
_observerObj = [[TestObject alloc]init];
[_observerObj addObserver:self forKeyPath:@"objName" options:NSKeyValueObservingOptionNew context:nil];

查看這兩個(gè)對(duì)象各自的isa指針可以發(fā)現(xiàn)灼狰,沒(méi)有添加observer的對(duì)象isa指針指向的類為TestObject宛瞄,添加observer的對(duì)象isa指針指向的類為NSKVONotifying_TestObject

isa.png

NSKVONotifying_TestObject和TestObject又是什么關(guān)系呢?


OC中類也同樣是一個(gè)對(duì)象交胚,它的isa指向創(chuàng)建這個(gè)類的類也就是元類(metaClass)份汗,它的superclass指向它的父類盈电。

我們不妨看看NSKVONotifying_TestObject這個(gè)類的父類是誰(shuí)。在此我們通過(guò)以下代碼來(lái)輸出它的父類:

//疑問(wèn):為什么我們?cè)谶@里使用[[_observerObj class] superclass]來(lái)獲取父類得到的是NSObject(筆者的解答會(huì)在文章末尾給出)
NSLog(@"%@",NSStringFromClass([[_observerObj valueForKey:@"isa"] superclass]));

控制臺(tái)輸出的類名為TestObject杯活。在此我們了解到NSKVONotifying_TestObject實(shí)際上為TestObject的子類匆帚。在我們?yōu)槟骋粋€(gè)對(duì)象添加observer時(shí),系統(tǒng)創(chuàng)建這個(gè)對(duì)象的類的子類旁钧,并將其命名為NSKVONotifying_父類名吸重,然后通過(guò)isa Swizzling將這個(gè)對(duì)象的isa指針替換。
那么NSKVONotifying_TestObject這個(gè)類又做了什么呢歪今?


我們將NSKVONotifying_TestObject的方法列表輸出

Class observerClass = [_observerObj valueForKey:@"isa"];
unsigned int outCount = 0;
Method * methodList = class_copyMethodList(observerClass, &outCount);
for (int i = 0; i < outCount; i ++) {
Method m = methodList[i];
SEL sel = method_getName(m);
NSLog(@"%@",NSStringFromSelector(sel));
}

NSKVONotifying_TestObject 中的方法.png

這一步我們可以看到NSKVONotifying_TestObject這個(gè)類重寫了父類中objName屬性的setter嚎幸。據(jù)此我們可以猜測(cè)NSKVONotifying_TestObject是通過(guò)setter觸發(fā)了 -(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context 。為了證明這一點(diǎn)筆者將會(huì)在下文中提供兩個(gè)示例寄猩。


示例1:

通過(guò)method swizzling來(lái)將NSKVONotifying_TestObject中的setter替換

Method oldMethod = class_getInstanceMethod([_observerObj valueForKey:@"isa"], NSSelectorFromString(@"setObjName:"));
Method newMethod = class_getInstanceMethod([self class], @selector(setObjName:));
method_setImplementation(oldMethod, method_getImplementation(newMethod));
NSLog(@"替換完成");
_observerObj.objName = @"我改變了";

替換的setter以及KVO回調(diào)實(shí)現(xiàn)如下:

- (void)setObjName:(NSString *)objName {
    NSLog(@"替換后的setter被調(diào)用");
}

- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context {
  if ([keyPath isEqualToString:@"objName"]) {
    NSLog(@"%@",_observerObj.objName);
  }
}

運(yùn)行結(jié)果如下圖:

替換setter后的運(yùn)行結(jié)果.png

從控制臺(tái)的輸出可以看到雖然執(zhí)行了我們替換后的setter但是 -(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context 并沒(méi)有被觸發(fā)嫉晶。


示例2:

我們?cè)诋?dāng)前類中定義以下屬性并添加observer。

@property (copy, nonatomic) NSString * myString;

[self addObserver:self forKeyPath:@"myString" options:NSKeyValueObservingOptionNew context:nil];

使用兩種方式賦值:

self.myString = @"我是通過(guò)setter賦值";
_myString = @"我是通過(guò)成員變量名賦值";

KVO回調(diào)方法實(shí)現(xiàn)如下:

- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context {
  if ([keyPath isEqualToString:@"myString"]) {
    NSLog(@"%@",_myString);
  }
}

控制臺(tái)打印結(jié)果如下圖:


兩種不同賦值方式控制臺(tái)輸出結(jié)果.png

可以看到我們通過(guò)setter賦值時(shí) KVO 回調(diào)執(zhí)行田篇,而直接使用成員變量名賦值時(shí)并沒(méi)有執(zhí)行车遂。


根據(jù)以上我們可以得出結(jié)論:KVO的觸發(fā)基于setter,如果我們?cè)跒楸O(jiān)聽(tīng)的屬性賦值時(shí)沒(méi)有通過(guò)setter賦值而是直接使用成員變量名來(lái)賦值KVO將不會(huì)被觸發(fā)斯辰。

對(duì)于上文中[[_observerObj class] superclass]獲取到的是NSObject筆者給出的解釋

在我們輸出NSKVONotifying_TestObject類的方法列表時(shí)其中出現(xiàn)了class方法舶担,在此筆者猜測(cè)這是由于NSKVONotifying_TestObject類重寫了class方法并且返回了superclass造成的。
為了證明筆者的猜測(cè)彬呻,筆者將NSKVONotifying_TestObject類中的class方法做了替換衣陶。

Method oldMethod = class_getInstanceMethod([_observerObj valueForKey:@"isa"], NSSelectorFromString(@"class"));
Method newMethod = class_getInstanceMethod([self class], @selector(kvo_class));
method_setImplementation(oldMethod, method_getImplementation(newMethod));

用于替換的kvo_class的實(shí)現(xiàn)

- (Class)kvo_class {
    return [super class];
}

此時(shí)我們調(diào)用_observerObj的class方法時(shí)得到的就是NSKVONotifying_TestObject


替換class方法后執(zhí)行[_observerObj class]得到的結(jié)果.png

如果我們將kvo_class的實(shí)現(xiàn)做以下修改會(huì)出現(xiàn)什么情況呢?

- (Class)kvo_class {
  return [super class].superclass;
}

return [[super class] superclass].png

參考

Objective-C對(duì)象模型及應(yīng)用
runtime Method

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末闸氮,一起剝皮案震驚了整個(gè)濱河市剪况,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌蒲跨,老刑警劉巖译断,帶你破解...
    沈念sama閱讀 206,602評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異或悲,居然都是意外死亡孙咪,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,442評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門巡语,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)翎蹈,“玉大人,你說(shuō)我怎么就攤上這事男公』缈埃” “怎么了?”我有些...
    開(kāi)封第一講書人閱讀 152,878評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)澄阳。 經(jīng)常有香客問(wèn)我拥知,道長(zhǎng),這世上最難降的妖魔是什么碎赢? 我笑而不...
    開(kāi)封第一講書人閱讀 55,306評(píng)論 1 279
  • 正文 為了忘掉前任低剔,我火速辦了婚禮,結(jié)果婚禮上揩抡,老公的妹妹穿的比我還像新娘户侥。我一直安慰自己镀琉,他們只是感情好峦嗤,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,330評(píng)論 5 373
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著屋摔,像睡著了一般烁设。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上钓试,一...
    開(kāi)封第一講書人閱讀 49,071評(píng)論 1 285
  • 那天装黑,我揣著相機(jī)與錄音,去河邊找鬼弓熏。 笑死恋谭,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的挽鞠。 我是一名探鬼主播疚颊,決...
    沈念sama閱讀 38,382評(píng)論 3 400
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼信认!你這毒婦竟也來(lái)了材义?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書人閱讀 37,006評(píng)論 0 259
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤嫁赏,失蹤者是張志新(化名)和其女友劉穎其掂,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體潦蝇,經(jīng)...
    沈念sama閱讀 43,512評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡款熬,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,965評(píng)論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了攘乒。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片华烟。...
    茶點(diǎn)故事閱讀 38,094評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖持灰,靈堂內(nèi)的尸體忽然破棺而出盔夜,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 33,732評(píng)論 4 323
  • 正文 年R本政府宣布喂链,位于F島的核電站返十,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏椭微。R本人自食惡果不足惜洞坑,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,283評(píng)論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望蝇率。 院中可真熱鬧迟杂,春花似錦、人聲如沸本慕。這莊子的主人今日做“春日...
    開(kāi)封第一講書人閱讀 30,286評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)锅尘。三九已至监氢,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間藤违,已是汗流浹背浪腐。 一陣腳步聲響...
    開(kāi)封第一講書人閱讀 31,512評(píng)論 1 262
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留顿乒,地道東北人议街。 一個(gè)月前我還...
    沈念sama閱讀 45,536評(píng)論 2 354
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像璧榄,于是被迫代替她去往敵國(guó)和親特漩。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,828評(píng)論 2 345

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

  • 轉(zhuǎn)至元數(shù)據(jù)結(jié)尾創(chuàng)建: 董瀟偉犹菱,最新修改于: 十二月 23, 2016 轉(zhuǎn)至元數(shù)據(jù)起始第一章:isa和Class一....
    40c0490e5268閱讀 1,681評(píng)論 0 9
  • [深入淺出Cocoa]詳解鍵值觀察(KVO)及其實(shí)現(xiàn)機(jī)理羅朝輝 (http://www.cppblog.com/k...
    Crazy2015閱讀 678評(píng)論 0 1
  • 上半年有段時(shí)間做了一個(gè)項(xiàng)目拾稳,項(xiàng)目中聊天界面用到了音頻播放,涉及到進(jìn)度條腊脱,當(dāng)時(shí)做android時(shí)候處理的不太好访得,由于...
    DaZenD閱讀 3,013評(píng)論 0 26
  • 終于把前面的base文件夾簡(jiǎn)簡(jiǎn)單單的看了一遍,終于可以回到正片上來(lái)了陕凹,保證不爛尾悍抑。 項(xiàng)目天天用yymodel解析數(shù)...
    充滿活力的早晨閱讀 1,358評(píng)論 1 0
  • 01 風(fēng)從天上來(lái) 帶著凜然的威勢(shì) 冷漠著螻蟻的內(nèi)心 02 云中仿佛隱匿著諸般恐懼 時(shí)刻準(zhǔn)備著擇人而噬 未知的結(jié)果叫...
    刀筆伐心閱讀 269評(píng)論 3 4