它們之間的關(guān)系可以這么解釋:一個(gè)類(Class)持有一個(gè)分發(fā)表,在運(yùn)行期分發(fā)消息,表中的每一個(gè)實(shí)體代表一個(gè)方法(Method),它的名字叫做選擇子(SEL),對(duì)應(yīng)著一種方法實(shí)現(xiàn)(IMP)。具體的分析如下
-
SEL:定義:
typedef struct objc_selector *SEL,代表方法的名稱。僅以名字來(lái)識(shí)別。翻譯成中文叫做選擇子或者選擇器,選擇子代表方法在 Runtime 期間的標(biāo)識(shí)符。為 SEL 類型,雖然 SEL 是objc_selector結(jié)構(gòu)體指針,但實(shí)際上它只是一個(gè) C 字符串。在類加載的時(shí)候,編譯器會(huì)生成與方法相對(duì)應(yīng)的選擇子,并注冊(cè)到 Objective-C 的 Runtime 運(yùn)行系統(tǒng)。不論兩個(gè)類是否存在依存關(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因此類方法定義時(shí),盡量不要用相同的名字,就算是變量類型不同也不行。否則會(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)前類實(shí)例的地址,如果是類則指向的是它的元類),作為消息的接受者;第二個(gè)參數(shù)代表方法的選擇子;... 代表可選參數(shù),前面的 id 代表返回值。Method:定義:
typedef struct objc_method *Method,Method對(duì)開(kāi)發(fā)者來(lái)說(shuō)是一種不透明的類型,被隱藏在我們平時(shí)書(shū)寫(xiě)的類或?qū)ο蟮姆椒ū澈?。它是一個(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類型為 SEL,前面提到過(guò)相同名字的方法即使在不同類中定義,它們的方法選擇器也相同。- 方法類型 method_types 是個(gè) char 指針,其實(shí)存儲(chǔ)著方法的參數(shù)類型和返回值類型,即是
Type Encoding編碼。method_imp指向方法的實(shí)現(xiàn),本質(zhì)上是一個(gè)函數(shù)的指針,就是前面講到的Implementation。
建了個(gè)群,群號(hào): 711315161,大家一起交流學(xué)習(xí)。