? ? ?runtime主要功能是消息發(fā)送,OC調(diào)用相關(guān)方法,在底層是通過objc_msgSend函數(shù)實現(xiàn)的,在Objective-C中消息的綁定是在代碼運行的時候進行的。
? ? ? ? 比如我們有一個類Receiver,類中定義了實例方法-(void)message;當我們OC代碼創(chuàng)建了Receiver的實例receiver時調(diào)用[receiver?message];編譯器在底層是調(diào)用了objc_msgSend(receiver, selector)函數(shù),我們在這里解釋下為什么調(diào)用objc_msgSend函數(shù)能正確調(diào)用我們OC的方法。首先我們先看下類和實例對象的原理圖:

? ? ? ? 當我們創(chuàng)建一個OC類的時候,編譯器會幫我們生成兩個關(guān)鍵的要素:1.指向父類的指針,2.一個包含方法的容器,該容器保存的元素記錄著方法提取器以及與之關(guān)聯(lián)的方法在內(nèi)存中的地址。當我們創(chuàng)建一個新的對象的時候,編譯器為這個對象分配內(nèi)存并且初始化,對象的第一個成員變量是一個指向該類的指針isa,通過這個指針我們可以找到對象的類,然后類中又有指向父類的指針,因此我們可以一層一層往上找直到頂級父類。
當我們向一個對象發(fā)送消息的時候,在OC中的體現(xiàn)是對象調(diào)用一個方法,編譯器后臺調(diào)用objc_msgSend函數(shù),該函數(shù)有兩個固定存在的參數(shù):1.消息的接收者(OC中調(diào)用方法的類的實例),2.方法選擇器,函數(shù)會根據(jù)對象的isa指針找到該類,然后從類中的方法容器里面查到對應(yīng)的方法選擇器,如果找不到就從父類里面找,一直找到頂級父類為止,如果找到了相應(yīng)的方法選擇器,然后就能找到與之關(guān)聯(lián)的方法實現(xiàn)的內(nèi)存地址,然后函數(shù)調(diào)用該方法,傳入對應(yīng)參數(shù)以及傳回返回值。為了提高消息轉(zhuǎn)發(fā)的速度,runtime會把調(diào)用過的方法緩存起來,該緩存是每個類相互獨立的,也能區(qū)分是父類的方法還是自身的方法。當下次消息發(fā)送的時候先檢查緩存區(qū),如果有對應(yīng)的緩存機直接調(diào)用緩存過的方法。