2022-03-09

? ? ? ? ? ? ? ? ? ? ? ? ? ?iOS Runtime 消息轉(zhuǎn)發(fā)機(jī)制原理

最近看到很多面試問道 runtime是個(gè)啥,咋實(shí)現(xiàn)的撒穷、具體實(shí)現(xiàn)的步驟勾笆,針對(duì)此類問題我做了一下整理,可能幫到小伙伴的可以看看桥滨。廢話不多說直接走起

我們?cè)陂_發(fā)過程中窝爪,經(jīng)常會(huì)遇到這樣的報(bào)錯(cuò),當(dāng)我們調(diào)用一個(gè)對(duì)象不存在的方法的時(shí)候

系統(tǒng)報(bào)錯(cuò) 提示如下錯(cuò)誤

"-[YBHomeHeaderCRView pushMessage:]: unrecognized selector sent to instance 0x7fc77cf0de40"

其實(shí)這種報(bào)錯(cuò)都是iOS消息轉(zhuǎn)發(fā)機(jī)制在無法響應(yīng)的情況后拋出的問題齐媒。那我們來深入的看一下這個(gè)消息機(jī)制是這么走的蒲每。

這么進(jìn)入的消息機(jī)制的

1.首先通過[YBHomeHeaderCRView new]對(duì)象的ISA指針找打它對(duì)應(yīng)的class。

2.首先在class的cache用查找是否有sendMessage方法喻括,沒有再去class的method list 查找邀杏,找到并將其加入到cache中,方便下次調(diào)用唬血。

3.如果沒有就去它的superclass里繼續(xù)查找望蜡。也是先cache,在methodlists拷恨。

4.一旦查找到對(duì)應(yīng)的函數(shù)方法脖律,通過函數(shù)指針I(yè)MP調(diào)轉(zhuǎn)到對(duì)應(yīng)的函數(shù)中去執(zhí)行。

5.如果一直沒有找到腕侄,就執(zhí)行消息轉(zhuǎn)發(fā)小泉。

消息機(jī)制原理

我們先看一下下面這個(gè)結(jié)構(gòu)圖芦疏,先對(duì)整個(gè)消息處理機(jī)制有一個(gè)初步的認(rèn)識(shí)

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

從結(jié)構(gòu)圖來看,消息轉(zhuǎn)發(fā)機(jī)制共分為3大步驟:

1.Method resolution 動(dòng)態(tài)方法解析處理階段

2.Fast forwarding 快速轉(zhuǎn)發(fā)階段

3.Normal forwarding 慢速轉(zhuǎn)發(fā)階段

如果想要不拋出unrecognized selector 的報(bào)錯(cuò)微姊,也就需要從這3步里面來做補(bǔ)救酸茴。

第一步?動(dòng)態(tài)方法解析處理階段

如果調(diào)用了對(duì)象方法首先會(huì)進(jìn)行+(BOOL)resolveInstanceMethod:(SEL)sel判斷

如果調(diào)用了類方法 首先會(huì)進(jìn)行 +(BOOL)resolveClassMethod:(SEL)sel判斷

如果YES則能接受消息,NO則不能接受消息 則就回進(jìn)入第二步

我們先調(diào)一下對(duì)象方法兢交,然后在resolveInstanceMethod進(jìn)行補(bǔ)救薪捍,

void pushMessage(id self, SEL _cmd, NSString *msg){

? ? NSLog(@"------%@",msg);

}

+(BOOL)resolveInstanceMethod:(SEL)sel{

? ? NSString *methodName =NSStringFromSelector(sel);

? ? if ([methodName isEqualToString:@"pushMessage:"]) {

? ? ? ? return class_addMethod(self, sel, (IMP)pushMessage, "v@:@");

? ? }

? ? return? NO;

}

經(jīng)過上面類型的補(bǔ)救,果然對(duì)象方法不在拋出異常了配喳,并且打印了數(shù)據(jù)

1

第二歩:快速轉(zhuǎn)發(fā)階段

如果在上一步的方法內(nèi)返回的為YES則能接受消息 NO不能接受消息 則進(jìn)入第二步酪穿,我們先把上面方法內(nèi)的處理方案注釋掉,讓消息轉(zhuǎn)發(fā)進(jìn)入第二步界逛。

我們新創(chuàng)建一個(gè)YBMessage類昆稿,里面聲明和實(shí)現(xiàn)pushMessage方法纺座,用來當(dāng)作備用響應(yīng)者息拜。

-(id)forwardingTargetForSelector:(SEL)aSelector{

? ? NSString *methodName =NSStringFromSelector(aSelector);

? ? if ([methodName isEqualToString:@"pushMessage:"]) {

? ? ? ? return [YBMessage new];

? ? }

? ? return [super forwardingTargetForSelector:aSelector];

}

因?yàn)橐粋€(gè)對(duì)象內(nèi)部可能還有其他可能響應(yīng)的對(duì)象,所以這個(gè)方法是轉(zhuǎn)發(fā)SEL去對(duì)象內(nèi)部的其他可以響應(yīng)該方法的對(duì)象净响。

第三歩 慢速轉(zhuǎn)發(fā)階段

如果第2步返回self或者nil,則說明沒有可以響應(yīng)的目標(biāo) 則進(jìn)入第三步少欺。第三步的消息轉(zhuǎn)發(fā)機(jī)制本質(zhì)上跟第二步是一樣的都是切換接受消息的對(duì)象,但是第三步切換響應(yīng)目標(biāo)更復(fù)雜一些馋贤,需要手動(dòng)將響應(yīng)方法切換給備用響應(yīng)對(duì)象赞别。

-(NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector{

? ? NSString*methodName =NSStringFromSelector(aSelector);

? ? if([methodNameisEqualToString:@"pushMessage:"]) {

? ? ? ? return [NSMethodSignature signatureWithObjCTypes:"v@:@"];

? ? }

? ? return [super methodSignatureForSelector:aSelector];

}

-(void)forwardInvocation:(NSInvocation *)anInvocation{

? ? SELsel = [anInvocationselector];

? ? YBMessage *msg = [YBMessage new];

? ? if([msg isEqualToString:sel]) {

? ? ? ? [anInvocationinvokeWithTarget:msg];

? ? }else{

? ? ? ? [superforwardInvocation:anInvocation];

? ? }

}

//例子"v@:@"

//v@:@ v:返回值類型void;@ id類型,執(zhí)行sel的對(duì)象;:SEL;@參數(shù)

如果前三個(gè)都沒有找到方法

- (void)doesNotRecognizeSelector:(SEL)aSelector{

? ? NSLog(@"找不到方法");

}

同時(shí),越往后面處理代價(jià)越高配乓,最好的情況是在第一步就處理消息仿滔,這樣runtime會(huì)在處理完后緩存結(jié)果,可以提高處理效率犹芹。

總結(jié):

在一個(gè)函數(shù)找不到時(shí)崎页,OC提供了三種方式去補(bǔ)救:

1、調(diào)用resolveInstanceMethod給個(gè)機(jī)會(huì)讓類添加這個(gè)實(shí)現(xiàn)這個(gè)函數(shù)

2腰埂、調(diào)用forwardingTargetForSelector讓別的對(duì)象去執(zhí)行這個(gè)函數(shù)

3飒焦、調(diào)用forwardInvocation(函數(shù)執(zhí)行器)靈活的將目標(biāo)函數(shù)以其他形式執(zhí)行。

如果都不中屿笼,調(diào)用doesNotRecognizeSelector拋出異常牺荠。

參考?iOS Runtime 消息轉(zhuǎn)發(fā)機(jī)制原理和實(shí)際用途

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市驴一,隨后出現(xiàn)的幾起案子休雌,更是在濱河造成了極大的恐慌,老刑警劉巖肝断,帶你破解...
    沈念sama閱讀 218,204評(píng)論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件挑辆,死亡現(xiàn)場(chǎng)離奇詭異例朱,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)鱼蝉,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,091評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門洒嗤,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人魁亦,你說我怎么就攤上這事渔隶。” “怎么了洁奈?”我有些...
    開封第一講書人閱讀 164,548評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵间唉,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我利术,道長(zhǎng)呈野,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,657評(píng)論 1 293
  • 正文 為了忘掉前任印叁,我火速辦了婚禮被冒,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘轮蜕。我一直安慰自己昨悼,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,689評(píng)論 6 392
  • 文/花漫 我一把揭開白布跃洛。 她就那樣靜靜地躺著率触,像睡著了一般。 火紅的嫁衣襯著肌膚如雪汇竭。 梳的紋絲不亂的頭發(fā)上葱蝗,一...
    開封第一講書人閱讀 51,554評(píng)論 1 305
  • 那天,我揣著相機(jī)與錄音细燎,去河邊找鬼两曼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛找颓,可吹牛的內(nèi)容都是我干的合愈。 我是一名探鬼主播,決...
    沈念sama閱讀 40,302評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼击狮,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼佛析!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起彪蓬,我...
    開封第一講書人閱讀 39,216評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤寸莫,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后档冬,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體膘茎,經(jīng)...
    沈念sama閱讀 45,661評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡桃纯,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,851評(píng)論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了披坏。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片态坦。...
    茶點(diǎn)故事閱讀 39,977評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖棒拂,靈堂內(nèi)的尸體忽然破棺而出伞梯,到底是詐尸還是另有隱情,我是刑警寧澤帚屉,帶...
    沈念sama閱讀 35,697評(píng)論 5 347
  • 正文 年R本政府宣布谜诫,位于F島的核電站,受9級(jí)特大地震影響攻旦,放射性物質(zhì)發(fā)生泄漏喻旷。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,306評(píng)論 3 330
  • 文/蒙蒙 一牢屋、第九天 我趴在偏房一處隱蔽的房頂上張望且预。 院中可真熱鬧,春花似錦伟阔、人聲如沸辣之。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,898評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至狮鸭,卻和暖如春合搅,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背歧蕉。 一陣腳步聲響...
    開封第一講書人閱讀 33,019評(píng)論 1 270
  • 我被黑心中介騙來泰國(guó)打工灾部, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人惯退。 一個(gè)月前我還...
    沈念sama閱讀 48,138評(píng)論 3 370
  • 正文 我出身青樓赌髓,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親催跪。 傳聞我的和親對(duì)象是個(gè)殘疾皇子锁蠕,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,927評(píng)論 2 355

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

  • 我們常常會(huì)聽說 Objective-C 是一門動(dòng)態(tài)語(yǔ)言,那么這個(gè)「動(dòng)態(tài)」表現(xiàn)在哪呢懊蒸?我想最主要的表現(xiàn)就是 Obje...
    Ethan_Struggle閱讀 2,195評(píng)論 0 7
  • OC Runtime之動(dòng)態(tài)方法解析和消息轉(zhuǎn)發(fā) 轉(zhuǎn)發(fā):鏈接 首先直接上代碼: @interface Person :...
    一凡_9b8b閱讀 351評(píng)論 0 0
  • 轉(zhuǎn)至元數(shù)據(jù)結(jié)尾創(chuàng)建: 董瀟偉荣倾,最新修改于: 十二月 23, 2016 轉(zhuǎn)至元數(shù)據(jù)起始第一章:isa和Class一....
    40c0490e5268閱讀 1,715評(píng)論 0 9
  • 設(shè)計(jì)模式是什么? 你知道哪些設(shè)計(jì)模式骑丸,并簡(jiǎn)要敘述舌仍? 設(shè)計(jì)模式是一種編碼經(jīng)驗(yàn)妒貌,就是用比較成熟的邏輯去處理某一種類型的...
    CoderBigBear閱讀 1,175評(píng)論 0 2
  • Runtime 理解 Runtime 又叫運(yùn)行時(shí),是一套底層的C語(yǔ)言API铸豁,其為iOS內(nèi)部的核心之一灌曙,我們平時(shí)編寫...
    愛讀書的強(qiáng)強(qiáng)閱讀 310評(píng)論 0 0