Objective-C是一門面向運(yùn)行時(shí)的語(yǔ)言浓恶,它盡可能的把一些編譯和鏈接時(shí)需要執(zhí)行的邏輯延遲到運(yùn)行時(shí),使Runtime可以按需要把消息重定向到合適的對(duì)象迈着,以實(shí)現(xiàn)動(dòng)態(tài)添加屬性和動(dòng)態(tài)方法交換等等蛔趴。
Objective-C在編譯階段并不會(huì)去綁定執(zhí)行的函數(shù),只有在運(yùn)行的時(shí)候才會(huì)到類和對(duì)象的方法列表中查找對(duì)應(yīng)的方法笨触。
OC中的方法查找
OC在進(jìn)行消息傳遞的時(shí)候,會(huì)先到緩存中以哈希查找的方式去查找緩存雹舀,若緩存命中芦劣,則執(zhí)行對(duì)應(yīng)方法。若未命中说榆,則從當(dāng)前類的方法列表中取查找持寄。
從當(dāng)前類的方法列表中查找(對(duì)于已完成排序的列表源梭,采用二分查找法去查找對(duì)應(yīng)的方法實(shí)現(xiàn)。對(duì)于沒有完成排序的算法稍味,采用一般遍歷查找法去查找對(duì)應(yīng)方法的實(shí)現(xiàn))废麻,若在當(dāng)前類方法列表中命中,則執(zhí)行模庐,否則再到父類方法列表中去查找烛愧,以重復(fù)這一過程。
OC中的消息傳遞
Objective-C中的消息發(fā)送是通過objc_msgSend(id, SEL,...)來實(shí)現(xiàn)的掂碱。Objective-C中的消息傳遞就是一次方法查找的過程怜姿。Objective-C調(diào)用一個(gè)方法,先去當(dāng)前類中查找該方法的imp疼燥,如果查找不到沧卢,則到父類中能夠去查找...以此類推,如果到根類中沒有查找到該方法醉者,則此次方法傳遞結(jié)束但狭。
OC中的消息轉(zhuǎn)發(fā)
Objective-C中消息轉(zhuǎn)發(fā)過程如下所示:
- 先調(diào)用resolveInstanceMethod,如果在這里使用runtime動(dòng)態(tài)添加對(duì)應(yīng)的方法撬即,并且返回YES立磁,消息就找到了響應(yīng)的對(duì)象,并將這個(gè)新增的方法添加到類的方法緩存列表
- 如果上面的方法返回NO的話剥槐,對(duì)象會(huì)調(diào)用forwardingTargetForSelector方法唱歧,以實(shí)現(xiàn)消息的轉(zhuǎn)發(fā),讓其他對(duì)象來處理這個(gè)消息粒竖。
- 如果以上兩個(gè)方法都沒有做處理颅崩,那么對(duì)象會(huì)執(zhí)行最后一個(gè)方法methodSignatureForSelector,提供一個(gè)有效的方法簽名蕊苗。若提供了有效的方法簽名挨摸,程序會(huì)通過forwardInvocation方法執(zhí)行簽名。若沒有提供方法簽名岁歉,觸發(fā)doesNotRecognizeSelector方法,觸發(fā)崩潰膝蜈。
resolveInstanceMethod
resolveInstanceMethod是Objective-C語(yǔ)言中一種動(dòng)態(tài)方法解析的接口锅移,是得我們可以在運(yùn)行時(shí)動(dòng)態(tài)的為一個(gè)selector提供實(shí)現(xiàn)。我們只需要實(shí)現(xiàn) +resolveInstanceMethod和+resolveClassMethod方法饱搏,并在其中為指定的selector提供實(shí)現(xiàn)即可(通過調(diào)用運(yùn)行時(shí)函數(shù)class_addMethod來添加)非剃。這兩個(gè)方法都是NSObject中的類方法,其原型為:
+ (BOOL)resolveClassMethod:(SEL)name;
+ (BOOL)resolveInstanceMethod:(SEL)name;
參數(shù)那么是需要被動(dòng)態(tài)解析的selector推沸;如果在該函數(shù)中為指定的selector提供實(shí)現(xiàn)备绽,無論返回YES還是NO券坞,編譯運(yùn)行都是正確的。如果在該函數(shù)內(nèi)并沒有真正的為selector提供實(shí)現(xiàn)肺素,如果返回YES恨锚,運(yùn)行會(huì)crash。其原理很簡(jiǎn)單倍靡,因?yàn)楫?dāng)前類既沒有為selector提供實(shí)現(xiàn)猴伶,又沒有實(shí)現(xiàn)消息轉(zhuǎn)發(fā),自然會(huì)crash塌西。
forwardingTargetForSelector
forwardingTargetForSelector是Objective-C語(yǔ)言中消息快速重定向的函數(shù)他挎。開發(fā)者可以在派生類中對(duì)其進(jìn)行重載,從而將無法處理的selector轉(zhuǎn)發(fā)給另一個(gè)對(duì)象捡需。
methodSignatureForSelector
methodSigntureForSelector的作用在在于為另一個(gè)類實(shí)現(xiàn)的消息創(chuàng)建一個(gè)有效的方法簽名办桨。如果沒有實(shí)現(xiàn)有效的方法簽名,程序就會(huì)崩潰
forwardInvocation
在返回有效的方法簽名的情況下站辉,當(dāng)前對(duì)象則會(huì)調(diào)用forwardInvocation方法呢撞,以完成消息的最終傳遞。