IOS下KVO使用過(guò)程中的陷阱

KVO,全稱(chēng)為Key-Value Observing,是iOS中的一種設(shè)計(jì)模式靖避,用于檢測(cè)對(duì)象的某些屬性的實(shí)時(shí)變化情況并作出響應(yīng)。網(wǎng)上廣為流傳普及的一個(gè)例子是利用KVO檢測(cè)股票價(jià)格的變動(dòng)比默,例如這里幻捏。這個(gè)例子作為掃盲入門(mén)還是可以的,但是當(dāng)應(yīng)用場(chǎng)景比較復(fù)雜時(shí)命咐,里面的一些細(xì)節(jié)還是需要改進(jìn)的篡九,里面有多個(gè)地方存在crash的危險(xiǎn)。本文旨在逐步遞進(jìn)深入地探討出一種目前比較健壯穩(wěn)定的KVO實(shí)現(xiàn)方案醋奠,彌補(bǔ)網(wǎng)上大部分教程的不足榛臼!

首先,假設(shè)我們的目標(biāo)是在一個(gè)UITableViewController內(nèi)對(duì)tableview的contentOffset進(jìn)行實(shí)時(shí)監(jiān)測(cè)窜司,很容易地使用KVO來(lái)實(shí)現(xiàn)為讽坏。

在初始化方法中加入:

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

在dealloc中移除KVO監(jiān)聽(tīng):

[_tableView removeObserver:self forKeyPath:@"contentOffset" context:nil]

添加默認(rèn)的響應(yīng)回調(diào)方法:

- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object

change:(NSDictionary *)change context:(void *)context

{

? ? ? ? ? ? [self doSomethingWhenContentOffsetChanges];

}

好了,KVO實(shí)現(xiàn)就到此完美結(jié)束了例证,拜拜路呜。。织咧。開(kāi)個(gè)玩笑胀葱,肯定沒(méi)這么簡(jiǎn)單的,這樣的代碼太粗糙了笙蒙,當(dāng)你在controller中添加多個(gè)KVO時(shí)抵屿,所有的回調(diào)都是走同上述函數(shù),那就必須對(duì)觸發(fā)回調(diào)函數(shù)的來(lái)源進(jìn)行判斷捅位。判斷如下:

- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object

change:(NSDictionary *)change context:(void *)context ?

{

? ? ? ? ? ?if (object == _tableView && [keyPath isEqualToString:@"contentOffset"]) {

? ? ? ? ? ? ? ? ? ?[self doSomethingWhenContentOffsetChanges];

? ? ? ? ? ? }?

}

你以為這樣就結(jié)束了嗎轧葛?答案是否定的!我們假設(shè)當(dāng)前類(lèi)(在例子中為UITableViewController)還有父類(lèi)艇搀,并且父類(lèi)也有自己綁定了一些其他KVO呢尿扯?我們看到,上述回調(diào)函數(shù)體中只有一個(gè)判斷焰雕,如果這個(gè)if不成立衷笋,這次KVO事件的觸發(fā)就會(huì)到此中斷了。但事實(shí)上矩屁,若當(dāng)前類(lèi)無(wú)法捕捉到這個(gè)KVO辟宗,那很有可能是在他的superClass爵赵,或者super-superClass...中,上述處理砍斷了這個(gè)鏈泊脐。合理的處理方式應(yīng)該是這樣的:

- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object

change:(NSDictionary *)change context:(void *)context

{

? ? ? ? ? ?if (object == _tableView && [keyPath isEqualToString:@"contentOffset"]) {

? ? ? ? ? ? ? ? ?[self doSomethingWhenContentOffsetChanges];

? ? ? ? ? ?} else {

? ? ? ? ? ?[super observeValueForKeyPath:keyPath ofObject:object change:change context:context];

? ? ? ? ? }

}

這樣就結(jié)束了嗎空幻?答案仍舊是否定的。潛在的問(wèn)題有可能出現(xiàn)在dealloc中對(duì)KVO的注銷(xiāo)上容客。KVO的一種缺陷(其實(shí)不能稱(chēng)為缺陷氛悬,應(yīng)該稱(chēng)為特性)是,當(dāng)對(duì)同一個(gè)keypath進(jìn)行兩次removeObserver時(shí)會(huì)導(dǎo)致程序crash耘柱,這種情況常常出現(xiàn)在父類(lèi)有一個(gè)kvo,父類(lèi)在dealloc中remove了一次棍现,子類(lèi)又remove了一次的情況下调煎。不要以為這種情況很少出現(xiàn)!當(dāng)你封裝framework開(kāi)源給別人用或者多人協(xié)作開(kāi)發(fā)時(shí)是有可能出現(xiàn)的己肮,而且這種crash很難發(fā)現(xiàn)士袄。不知道你發(fā)現(xiàn)沒(méi),目前的代碼中context字段都是nil谎僻,那能否利用該字段來(lái)標(biāo)識(shí)出到底kvo是superClass注冊(cè)的娄柳,還是self注冊(cè)的?

回答是可以的艘绍。我們可以分別在父類(lèi)以及本類(lèi)中定義各自的context字符串赤拒,比如在本類(lèi)中定義context為@"ThisIsMyKVOContextNotSuper";然后在dealloc中remove observer時(shí)指定移除的自身添加的observer。這樣iOS就能知道移除的是自己的kvo诱鞠,而不是父類(lèi)中的kvo挎挖,避免二次remove造成crash。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末航夺,一起剝皮案震驚了整個(gè)濱河市蕉朵,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌阳掐,老刑警劉巖始衅,帶你破解...
    沈念sama閱讀 217,084評(píng)論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異缭保,居然都是意外死亡汛闸,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,623評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門(mén)艺骂,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)蛉拙,“玉大人,你說(shuō)我怎么就攤上這事彻亲≡谐” “怎么了吮廉?”我有些...
    開(kāi)封第一講書(shū)人閱讀 163,450評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)畸肆。 經(jīng)常有香客問(wèn)我宦芦,道長(zhǎng),這世上最難降的妖魔是什么轴脐? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,322評(píng)論 1 293
  • 正文 為了忘掉前任调卑,我火速辦了婚禮,結(jié)果婚禮上大咱,老公的妹妹穿的比我還像新娘恬涧。我一直安慰自己,他們只是感情好碴巾,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,370評(píng)論 6 390
  • 文/花漫 我一把揭開(kāi)白布溯捆。 她就那樣靜靜地躺著,像睡著了一般厦瓢。 火紅的嫁衣襯著肌膚如雪提揍。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 51,274評(píng)論 1 300
  • 那天煮仇,我揣著相機(jī)與錄音劳跃,去河邊找鬼。 笑死浙垫,一個(gè)胖子當(dāng)著我的面吹牛刨仑,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播夹姥,決...
    沈念sama閱讀 40,126評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼贸人,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了佃声?” 一聲冷哼從身側(cè)響起艺智,我...
    開(kāi)封第一講書(shū)人閱讀 38,980評(píng)論 0 275
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎圾亏,沒(méi)想到半個(gè)月后十拣,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,414評(píng)論 1 313
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡志鹃,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,599評(píng)論 3 334
  • 正文 我和宋清朗相戀三年夭问,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片曹铃。...
    茶點(diǎn)故事閱讀 39,773評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡缰趋,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情秘血,我是刑警寧澤,帶...
    沈念sama閱讀 35,470評(píng)論 5 344
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響霞揉,放射性物質(zhì)發(fā)生泄漏适秩。R本人自食惡果不足惜政模,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,080評(píng)論 3 327
  • 文/蒙蒙 一耗式、第九天 我趴在偏房一處隱蔽的房頂上張望趁猴。 院中可真熱鬧娱挨,春花似錦跷坝、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,713評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)捞慌。三九已至,卻和暖如春嗅虏,著一層夾襖步出監(jiān)牢的瞬間皮服,已是汗流浹背龄广。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,852評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工净宵, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 47,865評(píng)論 2 370
  • 正文 我出身青樓已添,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親酝碳。 傳聞我的和親對(duì)象是個(gè)殘疾皇子矾踱,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,689評(píng)論 2 354

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