Objective-C runtime(二) 理解消息轉(zhuǎn)發(fā)機(jī)制

上一篇講到消息傳遞機(jī)制,那如果說(shuō),對(duì)象在收到無(wú)法解讀的消息之后會(huì)怎么做?
在開(kāi)發(fā)過(guò)程中,你可能會(huì)遇到這樣的錯(cuò)誤提示信息:

*** Terminating app due to uncaught exception 'NSInvalidArgumentException',
 reason: '-[ClassName messageName]: unrecognized selector sent to instance 0x7fd55850aaa0'

這說(shuō)明,你曾經(jīng)向某個(gè)對(duì)象發(fā)送過(guò)一條其無(wú)法解讀的消息,啟動(dòng)了消息轉(zhuǎn)發(fā)機(jī)制。最后由NSObject的"doesNotRecognizeSelector"方法拋出異常。

若想令類能理解某條消息,我們必須實(shí)現(xiàn)響應(yīng)代碼。但是,在編譯期間向類發(fā)送了無(wú)法解讀的消息并不會(huì)報(bào)錯(cuò),因?yàn)樵谶\(yùn)行期可以繼續(xù)向類中添加方法,所以編譯器在編譯時(shí)還無(wú)法確定類中到底會(huì)不會(huì)有某個(gè)方法實(shí)現(xiàn)。運(yùn)行期間,當(dāng)對(duì)象接收到無(wú)法解讀的消息后,就會(huì)啟動(dòng)『消息轉(zhuǎn)發(fā)』機(jī)制。在這個(gè)過(guò)程中,我們可以告訴對(duì)象,如何處理未知消息。

消息轉(zhuǎn)發(fā)分為三個(gè)階段。
第一階段,動(dòng)態(tài)方法解析

先征詢接收者,所屬的類,看其是否能動(dòng)態(tài)添加方法來(lái)處理當(dāng)前這個(gè)"unknown selector"。這叫做『動(dòng)態(tài)方法解析』(dynamic method resolution)。
對(duì)象在收到無(wú)法解讀的消息后,首先將調(diào)用其所屬類方法或者實(shí)例方法

+ (BOOL)resolveClassMethod:(SEL)sel
+ (BOOL)resolveInstanceMethod:(SEL)sel

該方法的參數(shù)就是對(duì)該類未知的選擇器(方法名),返回值表示這個(gè)類是否能新增一個(gè)方法用于處理該選擇器。

使用這種辦法的前提是:相關(guān)方法的實(shí)現(xiàn)代碼已經(jīng)寫(xiě)好,只等著運(yùn)行的時(shí)候通過(guò)class_addMethod函數(shù)動(dòng)態(tài)添加到類里面去。
這種方案常用來(lái)實(shí)現(xiàn)@dynamic屬性。

第二階段,備援接收者(replacement receiver)

當(dāng)前接收者還有第二次機(jī)會(huì)能處理未知的選擇器,在這一步,運(yùn)行期系統(tǒng)會(huì)詢問(wèn),是否可以把這條消息轉(zhuǎn)發(fā)給其他接收者。

- (id)forwardingTargetForSelector:(SEL)selector

如果對(duì)象實(shí)現(xiàn)了這個(gè)方法,并且返回非nil(不能是self,否則出現(xiàn)無(wú)限循環(huán)),則返回的對(duì)象成為新的接收者。

- (id)forwardingTargetForSelector:(SEL)aSelector {
    
    if ([_newReceiver respondsToSelector:aSelector]) {
        return _newReceiver;
    }
    return [super forwardingTargetForSelector:aSelector];
}

注意:這一步適用于把消息轉(zhuǎn)發(fā)到另一個(gè)能實(shí)現(xiàn)該消息的對(duì)象上。但是我們無(wú)法對(duì)消息進(jìn)行處理,比如操作消息的參數(shù)和返回值。

第三階段,完整的消息轉(zhuǎn)發(fā)

如果沒(méi)有『備援接收者』,則啟動(dòng)完整的消息轉(zhuǎn)發(fā)機(jī)制,運(yùn)行期系統(tǒng)會(huì)把與消息有關(guān)的全部細(xì)節(jié)都封裝到NSInvocation對(duì)象中,再給接收者最后一次機(jī)會(huì)。
首先創(chuàng)建NSInvocation對(duì)象,把與尚未處理的那條消息相關(guān)的全部細(xì)節(jié)都封于其中(選擇器,target,參數(shù))。
此步驟會(huì)調(diào)用下列方法來(lái)轉(zhuǎn)發(fā)消息:

- (void)forwardInvocation:(NSInvocation *)invocation

這個(gè)方法的實(shí)現(xiàn):改變調(diào)用目標(biāo),是消息在新目標(biāo)上得以調(diào)用即可。但是我們需要先重新寫(xiě)另外一個(gè)方法來(lái)獲取NSInvocation對(duì)象

- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector

消息轉(zhuǎn)發(fā)機(jī)制使用從這個(gè)方法中獲取的信息來(lái)創(chuàng)建NSInvocation對(duì)象。該方法為給定的selector提供了合適的方法簽名。

- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector {
    NSMethodSignature *signature = [super methodSignatureForSelector:aSelector];
    if (!signature) {
        if ([ReplacementReceiverObject instancesRespondToSelector:aSelector]) {
            signature = [ReplacementReceiverObject instanceMethodSignatureForSelector:aSelector];
        }
    }
    return signature;
}

- (void)forwardInvocation:(NSInvocation *)anInvocation {
    if ([ReplacementReceiverObject instancesRespondToSelector:anInvocation.selector]) {
        [anInvocation invokeWithTarget:receiver];
    }
}

然而,像上面這樣實(shí)現(xiàn)出來(lái)的方法與『備援接收者』方案所實(shí)現(xiàn)的方法等效,所以很少有人采用這么簡(jiǎn)單的實(shí)現(xiàn)方式。比較有用的實(shí)現(xiàn)方式為:在觸發(fā)消息前,先以某種方式改變消息內(nèi)容,比如追加另一個(gè)參數(shù),或者改換選擇器等等。

接收者在每一步中均有機(jī)會(huì)處理消息。步驟越往后,處理消息的代價(jià)就越大。最好能在第一步就處理完,這樣的話,運(yùn)行期系統(tǒng)就可以將此方法緩存起來(lái)了。如果這個(gè)類的實(shí)例稍后還收到同名選擇器,那么根本無(wú)須啟動(dòng)消息轉(zhuǎn)發(fā)流程。若想在第三步里把消息轉(zhuǎn)給備援接收者,那還不如把轉(zhuǎn)發(fā)操作提前到第二步。因?yàn)榈谌街皇切薷牧苏{(diào)用目標(biāo),這項(xiàng)改動(dòng)放在第二部執(zhí)行會(huì)更簡(jiǎn)單,不然的話,還得創(chuàng)建并處理完整的NSInvocation。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,786評(píng)論 6 534
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,656評(píng)論 3 419
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人,你說(shuō)我怎么就攤上這事。” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 176,697評(píng)論 0 379
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我,道長(zhǎng),這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 63,098評(píng)論 1 314
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 71,855評(píng)論 6 410
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 55,254評(píng)論 1 324
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,322評(píng)論 3 442
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 42,473評(píng)論 0 289
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 49,014評(píng)論 1 335
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 40,833評(píng)論 3 355
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 43,016評(píng)論 1 371
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,568評(píng)論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,273評(píng)論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 34,680評(píng)論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 35,946評(píng)論 1 288
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 51,730評(píng)論 3 393
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 48,006評(píng)論 2 374

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