Runtime窺探 (三)| 消息轉(zhuǎn)發(fā)

前言

其實(shí),重要的不是地圖榜苫,而是你的方向。所以翎冲,每條路都是一次冒險(xiǎn)垂睬,每個(gè)地點(diǎn)都是一場奇遇。

蒲柳之質(zhì)

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

上回書說道消息發(fā)送中最后沒有在類和父類/元類中找到方法時(shí)抗悍,程序會(huì)立即崩潰嗎驹饺?答案肯定不會(huì)立即崩潰。所以才引出來這篇消息轉(zhuǎn)發(fā)機(jī)制的文章缴渊,正常列表中找不到它會(huì)走到消息轉(zhuǎn)發(fā)系統(tǒng)中......

新建類:

//--------------------- Person.h ----------------------
#import <Foundation/Foundation.h>

@interface Person : NSObject

- (void)eatWith:(NSString *)name;
- (void)happy;


//--------------------- Person.m ----------------------
#import "Person.h"
#import <objc/message.h>

@implementation Person


- (void)eatWith:(NSString *)name{
    NSLog(@"實(shí)例方法正在和%@吃飯赏壹。。衔沼。蝌借。",name);
}
- (void)happy{
    NSLog(@"實(shí)例方法正在happy。指蚁。菩佑。。");
}

我們調(diào)用:

Person *p = [Person new];
[p performSelector:@selector(sleepWith:) withObject:@"Dely"];

結(jié)果:

-[Person sleepWith:]: unrecognized selector sent to instance 0x604000031260

程序崩潰凝化,報(bào)程序不認(rèn)識這個(gè)方法稍坯,也就是找不到這個(gè)方法的實(shí)現(xiàn)。

為什么會(huì)崩潰搓劫?因?yàn)槲覀冊谙l(fā)送時(shí)瞧哟,在類中的methodList中沒有找到這個(gè)方法列表袜蚕。而消息轉(zhuǎn)發(fā)機(jī)制就是處理找不到方法時(shí)的后續(xù)處理,為了更好的容錯(cuò)绢涡。

目的:用來處理消息的轉(zhuǎn)發(fā)和調(diào)用牲剃,給對象和消息更多的機(jī)會(huì)來完成成功的調(diào)用,而不是直接crash

消息轉(zhuǎn)發(fā)過程:

蘋果消息轉(zhuǎn)發(fā)官方文檔

消息轉(zhuǎn)發(fā)過程有三個(gè)階段:

  • 1.動(dòng)態(tài)方法解析
  • 2.快速消息轉(zhuǎn)發(fā)
  • 3.完整方法轉(zhuǎn)發(fā)

1.動(dòng)態(tài)方法解析

當(dāng)我們在消息發(fā)送中最后沒有在類和父類/元類中找到方法時(shí)雄可,首先會(huì)走到這個(gè)階段凿傅。
以下方法都在NSObject中定義
對應(yīng)的方法是

//方法是類方法時(shí),會(huì)調(diào)用這個(gè)函數(shù)
+ (BOOL)resolveClassMethod:(SEL)sel;

//方法是實(shí)例方法数苫,會(huì)調(diào)用這個(gè)函數(shù)
+ (BOOL)resolveInstanceMethod:(SEL)sel;

走到這個(gè)方法時(shí)可以利用class_addMethod給類添加方法提供了可能聪舒。

#pragma mark - --------1.動(dòng)態(tài)方法解析--------

//c方法
void sleepWith(id self,SEL _cmd , NSString *name){
    NSLog(@"和%@一起睡覺",name);
}

//oc方法
- (void) sleepWith:(NSString *)name{
     NSLog(@"和%@一起睡覺",name);
}

//IMP imp 方法的實(shí)現(xiàn),C方法的方法實(shí)現(xiàn)可以直接獲得虐急。
//如果是OC方法箱残,可以用+ (IMP)instanceMethodForSelector:(SEL)aSelector;獲得方法的實(shí)現(xiàn)。

//動(dòng)態(tài)解析方法
+ (BOOL)resolveInstanceMethod:(SEL)sel{
    if ([NSStringFromSelector(sel) isEqualToString:@"sleepWith:"]) {
        //        class_addMethod(self, sel, (IMP)happyWith, "v@:@");
        IMP imp = [self instanceMethodForSelector:NSSelectorFromString(@"happyWith:")];
        class_addMethod(self, sel, imp, "v@:@");
        return YES;
    }
    return [super resolveInstanceMethod:sel];
}

+ (BOOL)resolveClassMethod:(SEL)sel{
    return [super resolveClassMethod:sel];
}

上面Person重寫了```resolveInstanceMethod````方法

入?yún)ⅲ簊elector就是未被處理的方法

我們在方法中匹配了未被處理的方法名止吁,調(diào)用class_addMethod動(dòng)態(tài)添加了一個(gè)方法被辑,來實(shí)現(xiàn)缺少的方法,

class_addMethod中的參數(shù)const char *types就是我們上一篇講解的type encodings敬惦。IMP就是方法的實(shí)現(xiàn)變量

函數(shù)返回值:匹配到我們?nèi)鄙俚暮瘮?shù)就return YES;盼理,匹配不到就返回父類return [super resolveInstanceMethod:sel];

無論返回值為YES還是NO,都會(huì)進(jìn)入下一階段的快速消息轉(zhuǎn)發(fā)俄删。

此時(shí)再調(diào)用

Person *p = [Person new];
[p performSelector:@selector(sleepWith:) withObject:@"Dely"];

結(jié)果發(fā)現(xiàn)不會(huì)崩潰了:

和Dely一起睡覺

2.快速消息轉(zhuǎn)發(fā)

對應(yīng)的方法是

- (id)forwardingTargetForSelector:(SEL)aSelector;

這個(gè)階段的方法運(yùn)行時(shí)會(huì)詢問能否把消息轉(zhuǎn)給其他接收者處理宏怔,也就是此時(shí)系統(tǒng)給了個(gè)將這個(gè) SEL 轉(zhuǎn)給其他對象的機(jī)會(huì)。

#pragma mark - --------2.快速消息轉(zhuǎn)發(fā)--------
//將消息轉(zhuǎn)出給別的對象
- (id)forwardingTargetForSelector:(SEL)aSelector{
    NSLog(@"_cmd = %@",NSStringFromSelector(aSelector));
    
    if (aSelector== @selector(uppercaseString)) {
        //轉(zhuǎn)給字符串對象 
        return @"person";
    }
    return [super forwardingTargetForSelector:aSelector];
}

函數(shù)返回值:函數(shù)返回消息轉(zhuǎn)出的對象或者父類[super forwardingTargetForSelector:aSelector]

我們調(diào)用下面

//不會(huì)崩潰畴椰。 str = @"PERSON"
NSString *str =  [p performSelector:@selector(uppercaseString) withObject:nil];

注意:把消息轉(zhuǎn)出給別的對象執(zhí)行臊诊,是一對一的,也就是這個(gè)消息不能轉(zhuǎn)給很多個(gè)對象來執(zhí)行斜脂。

3.完整方法轉(zhuǎn)發(fā)

當(dāng)前面兩個(gè)階段都沒有處理也可以在這個(gè)階段來處理抓艳,
對應(yīng)的方法是

- (void)forwardInvocation:(NSInvocation *)anInvocation;
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector;

這是消息轉(zhuǎn)發(fā)的最后一個(gè)處理容錯(cuò)的機(jī)會(huì)。

  • methodSignatureForSelector 用于描述被轉(zhuǎn)發(fā)的消息秽褒,系統(tǒng)會(huì)調(diào)用methodSignatureForSelector:方法壶硅,嘗試獲得一個(gè)方法簽名威兜。如果獲取不到销斟,則直接調(diào)用doesNotRecognizeSelector拋出異常。如果能獲取椒舵,則返回非nil,然后創(chuàng)建一個(gè) NSlnvocation 并傳給forwardInvocation:

  • forwardInvocation 參數(shù) anInvocation 中包含未處理消息的各種信息(selector\target\參數(shù)...)蚂踊。在這個(gè)方法中,可以把 anInvocation 轉(zhuǎn)發(fā)給多個(gè)對象笔宿,因?yàn)?code>invokeWithTarget方法

#pragma mark - --------3.完整消息轉(zhuǎn)發(fā)--------
//完整消息轉(zhuǎn)發(fā)
-(void)forwardInvocation:(NSInvocation *)anInvocation{
    if (anInvocation.selector == @selector(testMethod)){
        ViewController *vc1 = [[ViewController alloc] init];
        ViewController *vc2 = [[ViewController alloc] init];
        [anInvocation invokeWithTarget:vc1];
        [anInvocation invokeWithTarget:vc2];
    }
}

//獲取方法簽名
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector {
    if(aSelector == @selector(testMethod)){
        return [NSMethodSignature signatureWithObjCTypes:"v@:"];
    }
    return nil;
} 

如果上述3個(gè)方法都沒有來處理這個(gè)消息犁钟,就會(huì)進(jìn)入 NSObject 的- (void)doesNotRecognizeSelector:(SEL)aSelector;方法中棱诱,拋出異常.

整個(gè)消息轉(zhuǎn)發(fā)的流程圖如下:

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

_objc_msgForward函數(shù)

//定義
_objc_msgForward(id _Nonnull receiver, SEL _Nonnull sel, ...) 

作用:用于消息轉(zhuǎn)發(fā)的皆愉。直接調(diào)用_objc_msgForward是非常危險(xiǎn)的事仰坦,用不好會(huì)直接導(dǎo)致程序Crash,但是使用得當(dāng)效果還是很炫的瘫想。

調(diào)用_objc_msgForward醋粟,將跳過查找 cache靡菇、Methodlist、消息轉(zhuǎn)發(fā)的第一個(gè)階段(動(dòng)態(tài)消息解析) 的過程米愿,直接觸發(fā)消息轉(zhuǎn)發(fā)的第2厦凤、3階段。就算類中有這個(gè)方法育苟,他也會(huì)告訴你沒有這個(gè)方法较鼓。而直接進(jìn)行消息轉(zhuǎn)發(fā)的2、3階段處理违柏。且用且珍惜

消息轉(zhuǎn)發(fā)總結(jié)

消息發(fā)送時(shí)博烂,在類和父類/元類中沒有找到方法時(shí),通過下面的3種消息轉(zhuǎn)發(fā)機(jī)制來進(jìn)行容錯(cuò)處理漱竖。

  • 1.調(diào)用resolveInstanceMethod給個(gè)機(jī)會(huì)動(dòng)態(tài)添加方法的實(shí)現(xiàn)
  • 2.調(diào)用forwardingTargetForSelector讓消息轉(zhuǎn)發(fā)給別的對象來執(zhí)行
  • 3.調(diào)用forwardInvocation和methodSignatureForSelector讓消息轉(zhuǎn)發(fā)給別的多個(gè)對象來執(zhí)行脖母。

如果都沒有處理這個(gè)消息,系統(tǒng)會(huì)調(diào)用doesNotRecognizeSelector拋出異常闲孤。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末谆级,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子讼积,更是在濱河造成了極大的恐慌肥照,老刑警劉巖,帶你破解...
    沈念sama閱讀 206,126評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件勤众,死亡現(xiàn)場離奇詭異舆绎,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)们颜,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,254評論 2 382
  • 文/潘曉璐 我一進(jìn)店門吕朵,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人窥突,你說我怎么就攤上這事努溃。” “怎么了阻问?”我有些...
    開封第一講書人閱讀 152,445評論 0 341
  • 文/不壞的土叔 我叫張陵梧税,是天一觀的道長。 經(jīng)常有香客問我,道長第队,這世上最難降的妖魔是什么哮塞? 我笑而不...
    開封第一講書人閱讀 55,185評論 1 278
  • 正文 為了忘掉前任,我火速辦了婚禮凳谦,結(jié)果婚禮上忆畅,老公的妹妹穿的比我還像新娘。我一直安慰自己尸执,他們只是感情好邻眷,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,178評論 5 371
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著剔交,像睡著了一般肆饶。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上岖常,一...
    開封第一講書人閱讀 48,970評論 1 284
  • 那天驯镊,我揣著相機(jī)與錄音,去河邊找鬼竭鞍。 笑死板惑,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的偎快。 我是一名探鬼主播冯乘,決...
    沈念sama閱讀 38,276評論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼晒夹!你這毒婦竟也來了裆馒?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 36,927評論 0 259
  • 序言:老撾萬榮一對情侶失蹤丐怯,失蹤者是張志新(化名)和其女友劉穎喷好,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體读跷,經(jīng)...
    沈念sama閱讀 43,400評論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡梗搅,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,883評論 2 323
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了效览。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片无切。...
    茶點(diǎn)故事閱讀 37,997評論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖丐枉,靈堂內(nèi)的尸體忽然破棺而出哆键,到底是詐尸還是另有隱情,我是刑警寧澤矛洞,帶...
    沈念sama閱讀 33,646評論 4 322
  • 正文 年R本政府宣布洼哎,位于F島的核電站,受9級特大地震影響沼本,放射性物質(zhì)發(fā)生泄漏噩峦。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,213評論 3 307
  • 文/蒙蒙 一抽兆、第九天 我趴在偏房一處隱蔽的房頂上張望识补。 院中可真熱鬧,春花似錦辫红、人聲如沸凭涂。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,204評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽切油。三九已至,卻和暖如春名惩,著一層夾襖步出監(jiān)牢的瞬間澎胡,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,423評論 1 260
  • 我被黑心中介騙來泰國打工娩鹉, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留攻谁,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 45,423評論 2 352
  • 正文 我出身青樓弯予,卻偏偏與公主長得像戚宦,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子锈嫩,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,722評論 2 345

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