前言
在面試過(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中看到:
根據(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的原因是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ì)其的介紹:
它的作用就是給一個(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ì)象爬虱。
從注釋看:對(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è)王圖:
總結(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