Objective-C消息轉(zhuǎn)發(fā)機制詳解

消息轉(zhuǎn)發(fā)機制的出現(xiàn)

? ? ? ? ? ? 回想一下姻乓,在我們?nèi)粘5某绦蜷_發(fā)中,是不是經(jīng)常會遇到下面這種情況:

自定義的Atom類
示例

? ? ? ? 我們自定義了一個類血筑,然后再程序中調(diào)用某個方法闰围,但是這個方法在該類中找不到相應實現(xiàn),系統(tǒng)提示我們unrecognized selector sent to instance榴徐,表明該對象無法解讀此消息秒梳,但作為一門高級的編程語言,難道Objective-C不做任何事就會讓程序崩潰掉嗎箕速,答案當然是...不可能酪碘!作為一門動態(tài)語言,Objective-C充分的利用了它的語言特性盐茎,在程序運行期間通過一些方法來給開發(fā)者機會阻斷程序這樣的崩潰兴垦,給對象和消息更多的機會來完成成功的調(diào)用,而這其中所用用到的技術就是消息轉(zhuǎn)發(fā)機制字柠。

什么是消息轉(zhuǎn)發(fā)機制

? ? ? ? 消息轉(zhuǎn)發(fā)機制探越,聽起來好像很厲害的一個概念,那么消息轉(zhuǎn)發(fā)機制到底是什么意思呢窑业,我的理解是消息轉(zhuǎn)發(fā)機制就是系統(tǒng)為了更好的解決對象收到無法解讀的消息這件事钦幔,然后自己創(chuàng)建出的一個模式,包含了一系列方法常柄,就像程序崩潰前的防守陣營一樣鲤氢,用來阻止程序直接走向崩潰。下面用一張圖來形象的展示一下消息轉(zhuǎn)發(fā)機制的構成:

消息轉(zhuǎn)發(fā)機制

? ? ? ?由圖中可以直觀的看到西潘,消息轉(zhuǎn)發(fā)機制主要包含三個步驟:

? ? ? ? 1:動態(tài)方法解析階段

? ? ? ? 2:備用接收者階段

? ? ? ? 3:完整消息轉(zhuǎn)發(fā)階段

? ? ? ? 如果程序經(jīng)過這三個階段之后都沒能解讀所調(diào)用的消息卷玉,那么程序會調(diào)走向最后的灰色提示,doesNotRecognizeSelector方法喷市,拋出異常相种,需要注意一點的是,上面三個步驟品姓,越往后寝并,處理消息的代價越大箫措。好啦,下面我們就來具體介紹一下這三個步驟吧衬潦!

動態(tài)方法解析階段

? ? ? ?這是消息轉(zhuǎn)發(fā)的第一階段蒂破,處在這個階段的時候,主要有兩個方法來幫你解決問題:

? ? ? ?1:+(BOOL)resolveClassMethod:(SEL)sel

? ? ? ?2:+(BOOL)resolveInstanceMethod:(SEL)sel

? ? ? 顧名思義别渔,當方法是類方法時調(diào)用1附迷,當方法為實例方法時,調(diào)用2哎媚。這個方法設計的目的是為了給類利用 class_addMethod 添加方法的機會喇伯。下面來看一下具體實現(xiàn)的例子:

示例

由代碼可知,在Atom類中我并未定義一個名為instanceTest的實例方法拨与,所以編譯器報了黃色的警告稻据,但我在Atom的實現(xiàn)文件中通過+(BOOL)resolveInstanceMethod:(SEL)sel這個方法在運行時動態(tài)的添加了方法的實現(xiàn),最終程序正常運行买喧。

? ? ? ? 關于class_addMethod(__unsafe_unretained Class cls, SEL name, IMP imp, const char *types)方法:

這里對于這個方法的參數(shù)做一點補充:

cls:被添加方法的目標類

name:新方法的選擇器

imp:實現(xiàn)這個方法的函數(shù)的地址

types:描述方法參數(shù)的數(shù)據(jù)類型的字符串捻悯,例如這個例子中我寫的id functionForInstance(id self,SEL _cmd),就對應“@@:”淤毛,這種表達方式叫Type Encodings今缚,是官方定義好的,更全的展示我在這貼兩張官方的圖:

type encodings
method encodings

備用接收者階段

? ? ? ? 這是消息轉(zhuǎn)發(fā)的第二階段低淡,如果第一階段沒有成功姓言,就會來到這個階段,在這個階段主要利用的方法是- (id)forwardingTargetForSelector:(SEL)aSelector蔗蹋,此時何荚,運行時詢問能否把消息轉(zhuǎn)給其他接收者處理,也就是此時系統(tǒng)給了個將這個 SEL 轉(zhuǎn)給其他對象的機會猪杭,具體示例如圖所示:

示例

我們新定義了一個類餐塘,名為Helper,在該類中我們實現(xiàn)了instanceTest實例方法皂吮,當系統(tǒng)運行時在Atom類中找不到方法的相應實現(xiàn)戒傻,于是來到- (id)forwardingTargetForSelector:(SEL)aSelector這個方法,請求備用者響應涮较,如果備用者能響應則將此消息轉(zhuǎn)給新對象執(zhí)行稠鼻。

完整消息轉(zhuǎn)發(fā)階段

? ? ? 如果前兩個階段都不能解決問題,系統(tǒng)就會來到最后這個第三階段狂票,完整消息轉(zhuǎn)發(fā)階段,這是消息轉(zhuǎn)發(fā)流程的最后一個環(huán)節(jié)熙暴,與第二階段不同闺属,此階段可以將消息轉(zhuǎn)發(fā)給多個對象慌盯,這個階段主要利用的方法是- (void)forwardInvocation:(NSInvocation *)anInvocation,但是注意一點,使用這個方法必須同時重寫- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector這個方法掂器,消息轉(zhuǎn)發(fā)機制使用從這個方法中獲取的信息來創(chuàng)建NSInvocation對象,具體示例如圖:

Helper實現(xiàn)文件
Helper2實現(xiàn)文件
Atom實現(xiàn)文件
示例

? ? ? 在這個例子中亚皂,我們新建了一個Helper2類,同樣實現(xiàn)了instanceTest實例方法国瓮,可以看到灭必,在這個例子中最后程序順利執(zhí)行了,同時Helper和Helper2都響應了這條消息乃摹,同時也能發(fā)現(xiàn)禁漓,最后的返回值是最后一個處理消息轉(zhuǎn)發(fā)對象的對應方法的返回值。

? ? ?補充一點孵睬,關于methodSignatureForSelector播歼,methodSignatureForSelector用于描述被轉(zhuǎn)發(fā)的消息,系統(tǒng)會調(diào)用methodSignatureForSelector:方法掰读,嘗試獲得一個方法簽名秘狞。如果獲取不到,則直接調(diào)用doesNotRecognizeSelector拋出異常蹈集。如果能獲取烁试,則返回非nil:創(chuàng)建一個 NSlnvocation 并傳給forwardInvocation:。

最后編輯于
?著作權歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末拢肆,一起剝皮案震驚了整個濱河市廓潜,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌善榛,老刑警劉巖辩蛋,帶你破解...
    沈念sama閱讀 217,657評論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異移盆,居然都是意外死亡据途,警方通過查閱死者的電腦和手機叙甸,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,889評論 3 394
  • 文/潘曉璐 我一進店門熔萧,熙熙樓的掌柜王于貴愁眉苦臉地迎上來贮缕,“玉大人,你說我怎么就攤上這事定嗓。” “怎么了层玲?”我有些...
    開封第一講書人閱讀 164,057評論 0 354
  • 文/不壞的土叔 我叫張陵铅碍,是天一觀的道長尘盼。 經(jīng)常有香客問我,道長午阵,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,509評論 1 293
  • 正文 為了忘掉前任氛魁,我火速辦了婚禮拖云,結果婚禮上,老公的妹妹穿的比我還像新娘尤筐。我一直安慰自己旬蟋,他們只是感情好冕碟,可當我...
    茶點故事閱讀 67,562評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著迎捺,像睡著了一般雹嗦。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上辅辩,一...
    開封第一講書人閱讀 51,443評論 1 302
  • 那天,我揣著相機與錄音玫锋,去河邊找鬼。 笑死谦炬,一個胖子當著我的面吹牛节沦,可吹牛的內(nèi)容都是我干的叫搁。 我是一名探鬼主播仿贬,決...
    沈念sama閱讀 40,251評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼茧泪,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了聋袋?” 一聲冷哼從身側(cè)響起嗜侮,我...
    開封第一講書人閱讀 39,129評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎啥容,沒想到半個月后锈颗,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,561評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡咪惠,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,779評論 3 335
  • 正文 我和宋清朗相戀三年击吱,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片遥昧。...
    茶點故事閱讀 39,902評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡覆醇,死狀恐怖朵纷,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情永脓,我是刑警寧澤袍辞,帶...
    沈念sama閱讀 35,621評論 5 345
  • 正文 年R本政府宣布,位于F島的核電站常摧,受9級特大地震影響搅吁,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜排宰,卻給世界環(huán)境...
    茶點故事閱讀 41,220評論 3 328
  • 文/蒙蒙 一似芝、第九天 我趴在偏房一處隱蔽的房頂上張望那婉。 院中可真熱鬧板甘,春花似錦、人聲如沸详炬。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,838評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽呛谜。三九已至在跳,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間隐岛,已是汗流浹背猫妙。 一陣腳步聲響...
    開封第一講書人閱讀 32,971評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留聚凹,地道東北人割坠。 一個月前我還...
    沈念sama閱讀 48,025評論 2 370
  • 正文 我出身青樓,卻偏偏與公主長得像妒牙,于是被迫代替她去往敵國和親彼哼。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 44,843評論 2 354

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

  • 轉(zhuǎn)至元數(shù)據(jù)結尾創(chuàng)建: 董瀟偉湘今,最新修改于: 十二月 23, 2016 轉(zhuǎn)至元數(shù)據(jù)起始第一章:isa和Class一....
    40c0490e5268閱讀 1,715評論 0 9
  • 消息發(fā)送和轉(zhuǎn)發(fā)流程可以概括為:消息發(fā)送(Messaging)是 Runtime 通過 selector 快速查找 ...
    lylaut閱讀 1,840評論 2 3
  • 我們常常會聽說 Objective-C 是一門動態(tài)語言敢朱,那么這個「動態(tài)」表現(xiàn)在哪呢?我想最主要的表現(xiàn)就是 Obje...
    Ethan_Struggle閱讀 2,193評論 0 7
  • 本文詳細整理了 Cocoa 的 Runtime 系統(tǒng)的知識摩瞎,它使得 Objective-C 如虎添翼拴签,具備了靈活的...
    lylaut閱讀 800評論 0 4
  • 轉(zhuǎn)載:http://yulingtianxia.com/blog/2014/11/05/objective-c-r...
    F麥子閱讀 733評論 0 2