Objective-C語言中方法的傳遞有二種:①Selector ② Blocks,本文主要說一下Selector,關於Blocks會在後續總結一下。
消息傳遞模型(Message Passing)是Objective-C語言的核心機制。在Objective-C中,沒有方法調用這種說法,只有消息傳遞。
在C++或Java中調用某個類的方法,在Objective-C中是給該類發送一個消息。在C++或Java裡,類與類的行為方法之間的關系非常緊密,一個方法必定屬於一個類,且於編譯時就已經綁定在一起,所以你不可能調用一個類裡沒有的方法。而在Objective-C中就比較簡單了,類和消息之間是松耦合的,方法調用只是向某個類發送一個消息,該類可以在運行時再確定怎麼處理接受到的消息。也就是說,一個類不保證一定會響應接收到的消息,如果收到了一個無法處理的消息,那麼程序既不會出錯也不或宕掉,它僅僅是什麼都不做,並返回一個nil【筆者添加:在編譯期是不出錯的,符合語義上的理解,但是runtime運行時的話,會崩潰】。這種設計本身也比較符合軟件的隱喻。(非常nice,從網上看到的,copy過來了)
很顯然,既然編譯器不定位方法,那麼只有運行期定位方法了,Objective-C又是怎麼去運行期定位方位的呢?
id objc_msgSend(id receiver, SEL selector, ...)【包含二個必要參數:receiver(接受者對象)、selector(方法選擇器)和一個未知參數(selector的參數列表)】
Objective-C就是通過上述方法來查找調用方法的~比如[itNoob cry]就被轉換成objc_msgSend(itNoob,cry),這裡receiver就是itNoob對象,selector就是cry選擇器,當然如果cry擁有參數的話,會同樣被轉換,如[itNoob cry:@"嗚嗚" AndSmile:@"嘻嘻"]會被轉換成objc_msgSend(itNoob,cry:AndSmile:,@"嗚嗚",@"嘻嘻"),類似如objc_msgSend(id receiver, SEL selector, parm1,parm2,...)。
objc_msgSend的動態綁定過程
那objc_msgSend的是如何查找方法的具體實現位置呢,從網上找了一下,如下:
編譯器構建每個類的時候,每個類必須包含二個必要的元素:
指向父類的指針一個調度表(dispatch table),調度表將類的selector與方法的實際內存地址關聯起來。我們知道每個對象都有一個isa指針,指向所屬類,通過這個isa指針可以找到對象的所屬類和所屬的父類...
查找過程如下:
當想一個對象發送消息的時候,先根據isa找到所屬的類,然後去查找該類的dispatch table,如果沒有找到,就去其父類中查找...如果找到了,就根據調度表中的內存地址調用該實現,如果最後一直沒有找到返回nil。