一.從objc_msgSend開始說(shuō)
[receiver message]
//會(huì)被編譯器轉(zhuǎn)化為
objc_msgSend(receiver, selector)
所以當(dāng)我們調(diào)用一個(gè)方法時(shí),會(huì)執(zhí)行的過(guò)程大致如下:
1.rutime系統(tǒng)會(huì)把方法調(diào)用轉(zhuǎn)化為消息發(fā)送握玛,并且把方法的調(diào)用者桥帆,和方法選擇器椅挣,當(dāng)做參數(shù)傳遞過(guò)去抛猖。
2.方法的調(diào)用者會(huì)通過(guò)isa 指針來(lái)找到所屬的類,然后在 cache 或者 methodLists 中查找該方法,找得到就跳到對(duì)應(yīng)的方法去執(zhí)行。
3.如果在類中沒(méi)找到該方法箭昵,則通過(guò)super_class 往上一級(jí)超類查找。如果一直找到 NSObject 都沒(méi)有找到該方法的話回季,可能就會(huì)觸發(fā)到消息轉(zhuǎn)發(fā)家制。
二. runtime 的術(shù)語(yǔ)的數(shù)據(jù)結(jié)構(gòu)
上面講的是執(zhí)行過(guò)程里面有一些術(shù)語(yǔ)接下來(lái)大概介紹下。
1.SEL
selector 是方法選擇器泡一,其實(shí)作用就和名字一樣颤殴,日常生活中,我們通過(guò)人名辨別誰(shuí)是誰(shuí)鼻忠。
Objective-C在編譯的時(shí)候涵但,會(huì)根據(jù)方法的名字,生成一個(gè)用 來(lái)區(qū)分這個(gè)方法的唯一的一個(gè)ID,這個(gè)ID就是SEL類型的贤笆。我們需要注意的是蝇棉,只要方法的名字相同讨阻,那么它們的ID都是相同的芥永。就是說(shuō),不管是超類還是子類钝吮,不管是有沒(méi)有超類和子類的關(guān)系埋涧,只要名字相同那么ID就是一樣的。
-(void)setWidth:(int)width奇瘦;
-(void)setWidth:(double)width棘催;
這樣的函數(shù)則被認(rèn)為是一種編譯錯(cuò)誤,而這最終導(dǎo)致了一個(gè)非常非常奇怪的Objective-C特色的函數(shù)命名:
-(void)setWidthIntValue:(int)width耳标;
-(void)setWidthDoubleValue:(double)width醇坝;
本質(zhì)上,SEL只是一個(gè)指向方法的指針次坡,它的存在只是為了加快方法的查詢速度呼猪。
2.Class
Class 其實(shí)是指向 objc_class 結(jié)構(gòu)體的指針
3.Method
Method 代表類中某個(gè)方法的類型
4.Ivar
Ivar 是表示成員變量的類型。
5.IMP
它就是一個(gè)函數(shù)指針砸琅,這是由編譯器生成的宋距。當(dāng)你發(fā)起一個(gè) ObjC 消息之后,最終它會(huì)執(zhí)行的那段代碼症脂,就是由這個(gè)函數(shù)指針指定的谚赎。而 IMP 這個(gè)函數(shù)指針就指向了這個(gè)方法的實(shí)現(xiàn)。
如果得到了執(zhí)行某個(gè)實(shí)例某個(gè)方法的入口诱篷,我們就可以繞開消息傳遞階段壶唤,直接執(zhí)行方法,這在后面 Cache 中會(huì)提到棕所。
你會(huì)發(fā)現(xiàn) IMP 指向的方法與 objc_msgSend 函數(shù)類型相同闸盔,參數(shù)都包含 id 和 SEL 類型。每個(gè)方法名都對(duì)應(yīng)一個(gè) SEL 類型的方法選擇器橙凳,而每個(gè)實(shí)例對(duì)象中的 SEL 對(duì)應(yīng)的方法實(shí)現(xiàn)肯定是唯一的蕾殴,通過(guò)一組 id和 SEL 參數(shù)就能確定唯一的方法實(shí)現(xiàn)地址。
而一個(gè)確定的方法也只有唯一的一組 id 和 SEL 參數(shù)岛啸。
6.Cache
Cache 為方法調(diào)用的性能進(jìn)行優(yōu)化钓觉,每當(dāng)實(shí)例對(duì)象接收到一個(gè)消息時(shí),它不會(huì)直接在 isa 指針指向的類的方法列表中遍歷查找能夠響應(yīng)的方法坚踩,因?yàn)槊看味家檎倚侍土说丛郑莾?yōu)先在 Cache 中查找。
Runtime 系統(tǒng)會(huì)把被調(diào)用的方法存到 Cache 中,如果一個(gè)方法被調(diào)用批幌,那么它有可能今后還會(huì)被調(diào)用础锐,下次查找的時(shí)候就會(huì)效率更高。就像計(jì)算機(jī)組成原理中 CPU 繞過(guò)主存先訪問(wèn) Cache 一樣荧缘。