它們之間的關(guān)系可以這么解釋?zhuān)阂粋€(gè)類(lèi)(Class)持有一個(gè)分發(fā)表僵缺,在運(yùn)行期分發(fā)消息胡桃,表中的每一個(gè)實(shí)體代表一個(gè)方法(Method),它的名字叫做選擇子(SEL)磕潮,對(duì)應(yīng)著一種方法實(shí)現(xiàn)(IMP)翠胰。具體的分析如下
-
SEL:定義:
typedef struct objc_selector *SEL
,代表方法的名稱(chēng)自脯。僅以名字來(lái)識(shí)別之景。翻譯成中文叫做選擇子或者選擇器,選擇子代表方法在 Runtime 期間的標(biāo)識(shí)符冤今。為 SEL 類(lèi)型闺兢,雖然 SEL 是objc_selector
結(jié)構(gòu)體指針,但實(shí)際上它只是一個(gè) C 字符串戏罢。在類(lèi)加載的時(shí)候屋谭,編譯器會(huì)生成與方法相對(duì)應(yīng)的選擇子,并注冊(cè)到 Objective-C 的 Runtime 運(yùn)行系統(tǒng)龟糕。不論兩個(gè)類(lèi)是否存在依存關(guān)系桐磁,只要他們擁有相同的方法名,那么他們的SEL都是相同的讲岁。比如我擂,有n個(gè)viewcontroller頁(yè)面衬以,每個(gè)頁(yè)面都有一個(gè)viewdidload,每個(gè)頁(yè)面的載入,肯定都是不盡相同的校摩。但是我們可以通過(guò)打印看峻,觀察發(fā)現(xiàn),這些viewdidload的SEL都是同一個(gè)SEL sel = @selector(methodName);
// 方法名字NSLog(@"address = %p",sel);
// log輸出為address = 0x1df807e29
因此類(lèi)方法定義時(shí)衙吩,盡量不要用相同的名字互妓,就算是變量類(lèi)型不同也不行。否則會(huì)引起重復(fù)坤塞,例如:-(void)setWidth:(int)width; -(void)setWidth:(double)width;
IMP:定義:
typedef id (*IMP)(id, SEL, ...)
冯勉,代表函數(shù)指針,即函數(shù)執(zhí)行的入口摹芙。該函數(shù)使用標(biāo)準(zhǔn)的 C 調(diào)用灼狰。第一個(gè)參數(shù)指向 self(它代表當(dāng)前類(lèi)實(shí)例的地址,如果是類(lèi)則指向的是它的元類(lèi))浮禾,作為消息的接受者交胚;第二個(gè)參數(shù)代表方法的選擇子;... 代表可選參數(shù)伐厌,前面的 id 代表返回值承绸。Method:定義:
typedef struct objc_method *Method裸影,Method
對(duì)開(kāi)發(fā)者來(lái)說(shuō)是一種不透明的類(lèi)型挣轨,被隱藏在我們平時(shí)書(shū)寫(xiě)的類(lèi)或?qū)ο蟮姆椒ū澈蟆K且粋€(gè)objc_method
結(jié)構(gòu)體指針轩猩,我們可以看到該結(jié)構(gòu)體中包含一個(gè)SEL和IMP卷扮,實(shí)際上相當(dāng)于在SEL和IMP之間作了一個(gè)映射。有了SEL均践,我們便可以找到對(duì)應(yīng)的IMP晤锹,從而調(diào)用方法的實(shí)現(xiàn)代碼。objc_method
的定義為:
/// Method
struct objc_method {
SEL method_name;
char *method_types;
IMP method_imp;
};
- 方法名
method_name
類(lèi)型為 SEL彤委,前面提到過(guò)相同名字的方法即使在不同類(lèi)中定義鞭铆,它們的方法選擇器也相同。- 方法類(lèi)型 method_types 是個(gè) char 指針焦影,其實(shí)存儲(chǔ)著方法的參數(shù)類(lèi)型和返回值類(lèi)型车遂,即是
Type Encoding
編碼。method_imp
指向方法的實(shí)現(xiàn)斯辰,本質(zhì)上是一個(gè)函數(shù)的指針舶担,就是前面講到的Implementation
。
建了個(gè)群彬呻,群號(hào): 711315161衣陶,大家一起交流學(xué)習(xí)柄瑰。