在Objective-C中蒙秒,message與方法的真正實(shí)現(xiàn)是在執(zhí)行階段綁定的苔埋,而非編譯階段贱迟。編譯器會(huì)將消息發(fā)送轉(zhuǎn)換成對(duì)objc_msgSend方法的調(diào)用塞关。
objc_msgSend方法含兩個(gè)必要參數(shù):receiver抬探、方法名(即:selector),如:
[receiver message];將被轉(zhuǎn)換為:objc_msgSend(receiver, selector);
objc_msgSend方法也能hold住message的參數(shù),如:
objc_msgSend(receiver, selector, arg1, arg2, …);
objc_msgSend方法會(huì)做按照順序進(jìn)行以下操作小压,以完成動(dòng)態(tài)綁定:
查找selector所指代的程序(方法的真正實(shí)現(xiàn))线梗。因?yàn)椴煌?lèi)對(duì)同一方法有不同的實(shí)現(xiàn),所以對(duì)方法的真正實(shí)現(xiàn)的查找依賴(lài)于receiver的類(lèi)
調(diào)用該實(shí)現(xiàn)怠益,并將一系列參數(shù)傳遞過(guò)去
將該實(shí)現(xiàn)的返回值作為自己的返回值仪搔,返回之
消息傳遞的關(guān)鍵是,編譯器構(gòu)建每個(gè)類(lèi)和對(duì)象時(shí)所采用的數(shù)據(jù)結(jié)構(gòu)蜻牢。每個(gè)類(lèi)都包含以下兩個(gè)必要元素:
一個(gè)指向父類(lèi)的指針
一個(gè)調(diào)度表(dispatch table)烤咧。該調(diào)度表將類(lèi)的selector與方法的實(shí)際內(nèi)存地址關(guān)聯(lián)起來(lái)
每個(gè)對(duì)象都有一個(gè)指向所屬類(lèi)的指針isa。通過(guò)該指針抢呆,對(duì)象可以找到它所屬的類(lèi)煮嫌,也就找到了其全部父類(lèi),如下圖所示:
當(dāng)向一個(gè)對(duì)象發(fā)送消息時(shí)抱虐,objc_msgSend方法根據(jù)對(duì)象的isa指針找到對(duì)象的類(lèi)昌阿,然后在類(lèi)的調(diào)度表(dispatch table)中查找selector。如果無(wú)法找到selector恳邀,objc_msgSend通過(guò)指向父類(lèi)的指針找到父類(lèi)懦冰,并在父類(lèi)的調(diào)度表(dispatch table)中查找selector,以此類(lèi)推直到NSObject類(lèi)谣沸。一旦查找到selector刷钢,objc_msgSend方法根據(jù)調(diào)度表的內(nèi)存地址調(diào)用該實(shí)現(xiàn)。通過(guò)這種方式鳄抒,message與方法的真正實(shí)現(xiàn)在執(zhí)行階段才綁定闯捎。
為了保證消息發(fā)送與執(zhí)行的效率,系統(tǒng)會(huì)將全部selector和使用過(guò)的方法的內(nèi)存地址緩存起來(lái)许溅。每個(gè)類(lèi)都有一個(gè)獨(dú)立的緩存瓤鼻,緩存包含有當(dāng)前類(lèi)自己的selector以及繼承自父類(lèi)的selector。查找調(diào)度表(dispatch table)前贤重,消息發(fā)送系統(tǒng)首先檢查receiver對(duì)象的緩存茬祷。
緩存命中的情況下,消息發(fā)送(messaging)比直接調(diào)用方法(function call)只慢一點(diǎn)點(diǎn)點(diǎn)點(diǎn)并蝗。
參考資料: