深入淺出理解消息的傳遞和轉(zhuǎn)發(fā)機(jī)制

前言

在面試過(guò)程中你也許會(huì)被問(wèn)到消息轉(zhuǎn)發(fā)機(jī)制。這篇文章就是對(duì)消息的轉(zhuǎn)發(fā)機(jī)制進(jìn)行一個(gè)梳理。主要包括什么是消息帮辟、靜態(tài)綁定/動(dòng)態(tài)綁定、消息的傳遞和消息的轉(zhuǎn)發(fā)玩焰。接下來(lái)開(kāi)發(fā)進(jìn)入正題由驹。

消息的解釋

在其他語(yǔ)言里面,我們可以用一個(gè)類去調(diào)用某個(gè)方法昔园,在OC里面蔓榄,這個(gè)方法就是消息。某個(gè)類調(diào)用一個(gè)方法就是向這個(gè)類發(fā)送一條消息默刚。舉個(gè)例子:

People *zhangSan = [[People alloc] init];
People *lisi = [[People alloc] init];
[zhangSan beFriendWith:lisi];

我們有個(gè)People的類甥郑,zhangSan這個(gè)實(shí)例發(fā)送了一條beFriendWith:的消息。你也許還看過(guò)這種調(diào)用方式:

[zhangSan performSelector:@selector(beFriendWith:) withObject:lisi];

其目和上面的一樣荤西,都是向zhangSan發(fā)送了一條beFriendWith:的消息壹若,傳人的參數(shù)都是lisi。
這里簡(jiǎn)單介紹一下SEL和IMP:

SEL:類成員方法的指針皂冰,但和C的函數(shù)指針還不一樣店展,函數(shù)指針直接保存了方法的地址,但是SEL只是方法編號(hào)秃流。
IMP:函數(shù)指針赂蕴,保存了方法地址。

我們叫@selector(beFriendWith:)為消息的選擇子或者選擇器舶胀。(A selector identifying the message to send)

靜態(tài)綁定/動(dòng)態(tài)綁定

所謂靜態(tài)綁定概说,就是在編譯期就能決定運(yùn)行時(shí)所調(diào)用的函數(shù),例如:

void printHello() {
    printf("Hello,world!\n");
}
void printGoodBye() {
    printf("Goodbye,world!\n");
}

void doTheThing(int type) {
    if (type == 0) {
        printHello();
    }else {
        printGoodBye();
    }
}

所謂動(dòng)態(tài)綁定嚣伐,就是在運(yùn)行期才能確定調(diào)用函數(shù):

void printHello() {
    printf("Hello,world!\n");
}
void printGoodBye() {
    printf("Goodbye,world!\n");
}
void doTheThing(int type) {
    void (*fnc)(void);
    if (type == 0) {
        fnc = printHello;
    }else {
        fnc = printGoodBye;
    }
    fnc();
}

在OC中糖赔,對(duì)象發(fā)送消息,就會(huì)使用動(dòng)態(tài)綁定機(jī)制來(lái)決定需要調(diào)用的方法轩端。其實(shí)底層都是C語(yǔ)言實(shí)現(xiàn)的函數(shù)放典,當(dāng)對(duì)象收到消息后,究竟調(diào)用那個(gè)方法完全決定于運(yùn)行期基茵,甚至你也可以直接在運(yùn)行時(shí)改變方法奋构,這些特性都使OC成為一門動(dòng)態(tài)語(yǔ)言。

消息的傳遞

先看一下一條簡(jiǎn)單的消息:

id returnValue = [someObject messageName:parameter];

其中:
someObject叫做接收者(receiver)拱层。
messageName叫做選擇器(selector)
選擇器和參數(shù)合起來(lái)成為消息(message)
當(dāng)編譯器看到這條消息弥臼,就會(huì)轉(zhuǎn)換成一條標(biāo)準(zhǔn)的C函數(shù):objc_msgSend,此時(shí)會(huì)變成:

id returnValue = objc_msgSend(someObject,@selector(messageName:),parameter);

objc_msgSend可以在objc里面的message.h中看到:


objc_msgSend

根據(jù)官方注釋可以看到:

When it encounters a method call, the compiler generates a call to one of the functions objc_msgSend, objc_msgSend_stret, objc_msgSendSuper, or objc_msgSendSuper_stret. Messages sent to an object’s superclass (using the super keyword) are sent using objc_msgSendSuper; other messages are sent using objc_msgSend. Methods that have data structures as return values are sent using objc_msgSendSuper_stret and objc_msgSend_stret.

它的作用是向一個(gè)實(shí)例類發(fā)送一個(gè)帶有簡(jiǎn)單返回值的message。是一個(gè)參數(shù)個(gè)數(shù)不定的函數(shù)根灯。當(dāng)遇到一個(gè)方法調(diào)用径缅,編譯器會(huì)生成一個(gè)objc_msgSend的調(diào)用掺栅,有:objc_msgSend_stret、objc_msgSendSuper或者是objc_msgSendSuper_stret纳猪。發(fā)送個(gè)父類的message會(huì)使用objc_msgSendSuper柿冲,其他的消息會(huì)使用objc_msgSend。如果方法的返回值是一個(gè)結(jié)構(gòu)體(structures)兆旬,那么就會(huì)使用objc_msgSendSuper_stret或者objc_msgSend_stret假抄。
第一個(gè)參數(shù)是:指向接收該消息的類的實(shí)例的指針
第二個(gè)參數(shù)是:要處理的消息的selector。
其他的就是要傳入的參數(shù)丽猬。
這樣消息派發(fā)系統(tǒng)就在接收者所屬類中查找器方法列表宿饱,如果找到和選擇器名稱相符的方法就跳轉(zhuǎn)其實(shí)現(xiàn)代碼,如果找不到脚祟,就再起父類找谬以,等找到合適的方法在跳轉(zhuǎn)到實(shí)現(xiàn)代碼。這里跳轉(zhuǎn)到實(shí)現(xiàn)代碼這一操作利用了尾遞歸優(yōu)化由桌。
如果該消息無(wú)法被該類或者其父類解讀为黎,就會(huì)開(kāi)始進(jìn)行消息轉(zhuǎn)發(fā)。

理解消息轉(zhuǎn)發(fā)機(jī)制(message forwarding)
動(dòng)態(tài)方法解析

不要把消息轉(zhuǎn)發(fā)機(jī)制想象得很難行您,其實(shí)看過(guò)下面的你就會(huì)發(fā)現(xiàn)铭乾,沒(méi)有那么難。
我們有的時(shí)候會(huì)遇到這樣的crash:


crash

我們都知道crash的原因是People沒(méi)有g(shù)otoschool這個(gè)方法娃循,但是你調(diào)用了該方法炕檩,所以會(huì)產(chǎn)生NSInvalidArgumentException,reason:

-[People gotoschool]: unrecognized selector sent to instance 0x1d4201780'

接下來(lái)讓我們看看從發(fā)送消息到此crash的過(guò)程捌斧。前面消息的傳遞沒(méi)有成功找到實(shí)現(xiàn)笛质,所以會(huì)走到消息轉(zhuǎn)發(fā)里面,我先在People類里面實(shí)現(xiàn)了這樣一個(gè)方法:

void gotoSchool(id self,SEL _cmd,id value) {
    printf("go to school");
}
//對(duì)象在收到無(wú)法解讀的消息后捞蚂,首先將調(diào)用所屬類的該方法妇押。
+ (BOOL)resolveInstanceMethod:(SEL)sel {
    NSString *selectorString = NSStringFromSelector(sel);
    if ([selectorString isEqualToString:@"gotoschool"]) {
        class_addMethod(self, sel, (IMP)gotoSchool, "@@:");
    }
    return [super resolveInstanceMethod:sel];
}

然后再次運(yùn)行程序,你會(huì)發(fā)現(xiàn)沒(méi)有crash了姓迅,而且順利打印出來(lái)"go to school"敲霍。
這個(gè)是什么個(gè)情況呢?先看看這個(gè)方法:

+ (BOOL)resolveInstanceMethod:(SEL)sel OBJC_AVAILABLE(10.5, 2.0, 9.0, 1.0, 2.0);
+ (BOOL)resolveClassMethod:(SEL)sel OBJC_AVAILABLE(10.5, 2.0, 9.0, 1.0, 2.0);

這個(gè)方法是objc里面NSObject.h里面的方法队贱。從字面理解就是處理實(shí)例方法(處理類方法)色冀。下面是對(duì)其的介紹:


resolveInstanceMethod/forwardingTargetForSelector:

它的作用就是給一個(gè)實(shí)例方法(給定的選擇器)動(dòng)態(tài)提供一個(gè)實(shí)現(xiàn)。注釋也提供了一個(gè)demo告訴我們?nèi)绾蝿?dòng)態(tài)添加實(shí)現(xiàn)柱嫌。
也就是說(shuō)當(dāng)消息傳遞無(wú)法處理的時(shí)候,首先會(huì)看一下所屬類屯换,是否能動(dòng)態(tài)添加方法编丘,以處理當(dāng)前未知的選擇子与学。這個(gè)過(guò)程叫做“動(dòng)態(tài)方法解析”(dynamic method resolution)。
這里我在動(dòng)態(tài)方法解析這里動(dòng)態(tài)添加了實(shí)現(xiàn)嘉抓,然后程序就不會(huì)崩潰啦索守。
如果是類方法,就調(diào)用resolveClassMethod:方法進(jìn)行操作抑片,和上面的resolveInstanceMethod一樣的處理方式卵佛。
這里還用到了calss_addMethod,后面會(huì)單獨(dú)寫篇博客對(duì)其介紹敞斋。感興趣的可以先自行查看API截汪。

備援接收者

當(dāng)動(dòng)態(tài)方法解析沒(méi)有實(shí)現(xiàn)或者無(wú)法處理的時(shí)候,就會(huì)執(zhí)行

- (id)forwardingTargetForSelector:(SEL)aSelector OBJC_AVAILABLE(10.5, 2.0, 9.0, 1.0, 2.0);

這個(gè)方法也是objc里面NSObject.h里面的方法植捎。我對(duì)People進(jìn)行了如下處理:

- (id)forwardingTargetForSelector:(SEL)aSelector {
    NSString *selectorString = NSStringFromSelector(aSelector);
    if ([selectorString isEqualToString:@"gotoschool"]) {
        return self.student;
    }
    return nil;
    
}

我在People里面添加了一個(gè)Student類實(shí)例衙解,然后實(shí)現(xiàn)了forwardingTargetForSelector:方法。然后運(yùn)行焰枢,奇跡地發(fā)現(xiàn)程序也沒(méi)有崩潰蚓峦。該方法的作用是(上圖也有介紹):
返回一個(gè)對(duì)未識(shí)別消息處理的對(duì)象。如果實(shí)現(xiàn)了該方法济锄,并且該方法沒(méi)有返回nil暑椰,那么這個(gè)返回的對(duì)象就會(huì)作為新的接收對(duì)象,這個(gè)未知的消息將會(huì)被新對(duì)象處理荐绝。通過(guò)此方案干茉,我們可以用組合來(lái)模擬多重繼承的某些特性,比如我返回多個(gè)類的組合很泊,那么就像繼承多個(gè)類一樣進(jìn)行處理角虫。在對(duì)外調(diào)用者來(lái)說(shuō),好像就是該對(duì)象親自處理的這些消息委造。

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

當(dāng)動(dòng)態(tài)方法解析和備援接收者都沒(méi)有進(jìn)行處理的話戳鹅,就會(huì)執(zhí)行:

- (void)forwardInvocation:(NSInvocation *)anInvocation OBJC_SWIFT_UNAVAILABLE("");

這個(gè)方法也是objc里面NSObject.h里面的方法,我對(duì)People進(jìn)行如下處理:

- (void)forwardInvocation:(NSInvocation *)anInvocation {
    NSLog(@"%@ can't handle by People",NSStringFromSelector([anInvocation selector]));
}
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector {
    NSMethodSignature *sign = [NSMethodSignature signatureWithObjCTypes:"@@:"];
    return sign;
}

再次運(yùn)行程序昏兆,發(fā)現(xiàn)程序沒(méi)有崩潰枫虏,只不過(guò)打印出來(lái)了“gotoschool can't handle by People”。
forwardInvocation:方法是將消息轉(zhuǎn)發(fā)給其他對(duì)象爬虱。


forwardInvocation:

從注釋看:對(duì)一個(gè)你的對(duì)象不識(shí)別的消息進(jìn)行相應(yīng)隶债,你必須重寫methodSignatureForSelector:方法,該方法返回一個(gè)NSMethodSIgnature對(duì)象跑筝,該對(duì)象包含了給定選擇器所標(biāo)識(shí)方法的描述死讹。主要包含返回值的信息和參數(shù)信息。
實(shí)現(xiàn)forwardInvocation:方法時(shí)曲梗,若發(fā)現(xiàn)調(diào)用的message不是由本類處理赞警,則續(xù)調(diào)用超類的同名方法妓忍。這樣所有父類均有機(jī)會(huì)處理此消息,直到NSObject愧旦。如果最后調(diào)用了NSObject的方法世剖,那么該方法就會(huì)調(diào)用“doesNotRecognizerSelector:”,拋出異常笤虫,標(biāo)明選擇器最終未能得到處理旁瘫。也就是上面的crash:NSInvalidArgumentException。
至此琼蚯,真?zhèn)€消息轉(zhuǎn)發(fā)全流程結(jié)束酬凳。
上一個(gè)王圖:


消息轉(zhuǎn)發(fā)全流程

總結(jié)

接收者在每一步都有機(jī)會(huì)對(duì)未知消息進(jìn)行處理,一句話:越早處理越好凌停。如果能在第一步做完粱年,就不進(jìn)行其他操作,因?yàn)閯?dòng)態(tài)方法解析會(huì)將此方法緩存罚拟。如果動(dòng)態(tài)方法解析不了台诗,就放到第二步備援接收者,因?yàn)榈谌竭€要?jiǎng)?chuàng)建完整的NSInvocation赐俗。
在完整來(lái)一遍:
Q:說(shuō)一下你理解的消息轉(zhuǎn)發(fā)機(jī)制拉队?
A:
先會(huì)調(diào)用objc_msgSend方法,首先在Class中的緩存查找IMP阻逮,沒(méi)有緩存則初始化緩存粱快。如果沒(méi)有找到,則向父類的Class查找叔扼。如果一直查找到根類仍舊沒(méi)有實(shí)現(xiàn)事哭,則執(zhí)行消息轉(zhuǎn)發(fā)。
1瓜富、調(diào)用resolveInstanceMethod:方法鳍咱。允許用戶在此時(shí)為該Class動(dòng)態(tài)添加實(shí)現(xiàn)。如果有實(shí)現(xiàn)了与柑,則調(diào)用并返回YES谤辜,重新開(kāi)始o(jì)bjc_msgSend流程。這次對(duì)象會(huì)響應(yīng)這個(gè)選擇器价捧,一般是因?yàn)樗呀?jīng)調(diào)用過(guò)了class_addMethod丑念。如果仍沒(méi)有實(shí)現(xiàn),繼續(xù)下面的動(dòng)作结蟋。
2脯倚、調(diào)用forwardingTargetForSelector:方法,嘗試找到一個(gè)能響應(yīng)該消息的對(duì)象椎眯。如果獲取到挠将,則直接把消息轉(zhuǎn)發(fā)給它胳岂,返回非nil對(duì)象编整。否則返回nil舔稀,繼續(xù)下面的動(dòng)作。注意這里不要返回self掌测,否則會(huì)形成死循環(huán)内贮。
3、調(diào)用methodSignatureForSelector:方法汞斧,嘗試獲得一個(gè)方法簽名夜郁。如果獲取不到,則直接調(diào)用doesNotRecognizeSelector拋出異常粘勒。如果能獲取竞端,則返回非nil;傳給一個(gè)NSInvocation并傳給forwardInvocation:。
4庙睡、調(diào)用forwardInvocation:方法事富,將第三步獲取到的方法簽名包裝成Invocation傳入,如何處理就在這里面了乘陪,并返回非nil统台。
5、調(diào)用doesNotRecognizeSelector:啡邑,默認(rèn)的實(shí)現(xiàn)是拋出異常贱勃。如果第三步?jīng)]能獲得一個(gè)方法簽名,執(zhí)行該步驟 谤逼。

另附相關(guān)雜亂代碼(里面有動(dòng)態(tài)方法解析demo)贵扰。
轉(zhuǎn)載請(qǐng)注明來(lái)源:http://www.cnblogs.com/zhanggui/p/7731394.html

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市流部,隨后出現(xiàn)的幾起案子戚绕,更是在濱河造成了極大的恐慌,老刑警劉巖贵涵,帶你破解...
    沈念sama閱讀 219,366評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件列肢,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡宾茂,警方通過(guò)查閱死者的電腦和手機(jī)瓷马,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,521評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)跨晴,“玉大人欧聘,你說(shuō)我怎么就攤上這事《伺瑁” “怎么了怀骤?”我有些...
    開(kāi)封第一講書人閱讀 165,689評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵费封,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我蒋伦,道長(zhǎng)弓摘,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書人閱讀 58,925評(píng)論 1 295
  • 正文 為了忘掉前任痕届,我火速辦了婚禮韧献,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘研叫。我一直安慰自己锤窑,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,942評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布嚷炉。 她就那樣靜靜地躺著渊啰,像睡著了一般。 火紅的嫁衣襯著肌膚如雪申屹。 梳的紋絲不亂的頭發(fā)上绘证,一...
    開(kāi)封第一講書人閱讀 51,727評(píng)論 1 305
  • 那天,我揣著相機(jī)與錄音独柑,去河邊找鬼迈窟。 笑死,一個(gè)胖子當(dāng)著我的面吹牛忌栅,可吹牛的內(nèi)容都是我干的车酣。 我是一名探鬼主播,決...
    沈念sama閱讀 40,447評(píng)論 3 420
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼索绪,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼湖员!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起瑞驱,我...
    開(kāi)封第一講書人閱讀 39,349評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤娘摔,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后唤反,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體凳寺,經(jīng)...
    沈念sama閱讀 45,820評(píng)論 1 317
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,990評(píng)論 3 337
  • 正文 我和宋清朗相戀三年彤侍,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了肠缨。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,127評(píng)論 1 351
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡盏阶,死狀恐怖晒奕,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤脑慧,帶...
    沈念sama閱讀 35,812評(píng)論 5 346
  • 正文 年R本政府宣布魄眉,位于F島的核電站,受9級(jí)特大地震影響闷袒,放射性物質(zhì)發(fā)生泄漏坑律。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,471評(píng)論 3 331
  • 文/蒙蒙 一霜运、第九天 我趴在偏房一處隱蔽的房頂上張望脾歇。 院中可真熱鬧蒋腮,春花似錦淘捡、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書人閱讀 32,017評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至作彤,卻和暖如春膘魄,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背竭讳。 一陣腳步聲響...
    開(kāi)封第一講書人閱讀 33,142評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工创葡, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人绢慢。 一個(gè)月前我還...
    沈念sama閱讀 48,388評(píng)論 3 373
  • 正文 我出身青樓灿渴,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親胰舆。 傳聞我的和親對(duì)象是個(gè)殘疾皇子骚露,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,066評(píng)論 2 355

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