可能大家一直看到有許多朋友在Runtime相關(guān)文章中介紹IMP指針的概念淌铐,那么IMP究竟有什么實(shí)際作用呢?讓我們先從一個(gè)函數(shù)看起來蔫缸。
Method Swizzling
如果對(duì)Runtime有一定了解的話腿准,一定聽說過或者用過這個(gè)函數(shù):
void method_exchangeImplementations(Method m1, Method m2)
它通常叫做method swizzling,算是ObjC的"黑魔法"了捂龄,作用就是在程序運(yùn)行期間動(dòng)態(tài)的給兩個(gè)方法互換實(shí)現(xiàn)释涛,比如有這樣一種使用場(chǎng)景:
我們的程序中有許多個(gè)ViewController,我想在對(duì)項(xiàng)目改動(dòng)最小的情況下倦沧,在當(dāng)每個(gè)Controller執(zhí)行完ViewDidLoad以后就在控制臺(tái)把自己的名字打印出來唇撬,方便我去做調(diào)試或者了解項(xiàng)目結(jié)構(gòu)。
有許多朋友會(huì)這樣說展融,讓所有控制器都繼承一個(gè)BaseController不就可以了嗎窖认?我在這里要解釋一下這樣做的缺點(diǎn):假如你的項(xiàng)目里有許多Controller的話,你就需要把項(xiàng)目里凡是沒有繼承自BaseController的每個(gè)Controller都做一次修改了,而且隨意更改層級(jí)結(jié)構(gòu)會(huì)發(fā)生意想不到的錯(cuò)誤扑浸。
其實(shí)我們的目的
就是重寫ViewDidLoad的方法烧给,并在他的方法最后加上幾句Log,所以我們需要給UIViewController建立一個(gè)category喝噪,因?yàn)槲覀冎来〉眨绻贑atagory中重寫一個(gè)方法,就會(huì)覆蓋它的原有方法實(shí)現(xiàn)酝惧,但是榴鼎,這樣做以后就沒有辦法調(diào)用系統(tǒng)原有的方法,因?yàn)樵谝粋€(gè)方法里調(diào)用自己的方法會(huì)是一個(gè)死循環(huán)晚唇。所以我們的解決辦法就是巫财,另外寫一個(gè)方法來和viewDidLoad“交換”,這樣外部調(diào)用viewDidLoad就會(huì)調(diào)到新建的這個(gè)方法中哩陕,同樣平项,我們調(diào)用新建的方法就會(huì)調(diào)用到系統(tǒng)的viewDidLoad中了。
IMP指針
其實(shí)悍及,還有一種更加簡(jiǎn)單的方法可以讓我們辦到相同的目的闽瓢,運(yùn)用IMP指針,IMP就是Implementation的縮寫并鸵,顧名思義鸳粉,它是指向一個(gè)方法實(shí)現(xiàn)的指針,每一個(gè)方法都有一個(gè)對(duì)應(yīng)的IMP园担,所以届谈,我們可以直接調(diào)用方法的IMP指針,來避免方法調(diào)用死循環(huán)的問題弯汰。
調(diào)用一個(gè)IMP的方式和調(diào)用普通C函數(shù)相同艰山,比如:
id returnObjc = someIMP(objc,SEL,params...);
不過如果你的項(xiàng)目沒有做其他配置的話這樣調(diào)用編譯器是不會(huì)通過的,我們來看一下先它的定義:
if !OBJC_OLD_DISPATCH_PROTOTYPES
typedef void (*IMP)(void /* id, SEL, ... */ );
else
typedef id (*IMP)(id, SEL, ...);
endif
在默認(rèn)情況下你的工程是打開這個(gè)配置的
這種情況下IMP被定義為無參數(shù)無返回值的函數(shù)咏闪。所以你需要到工程里搜索到這個(gè)選項(xiàng)并把它關(guān)閉曙搬。這樣的麻煩就是,每次使用鸽嫂,你都需要修改工程配置纵装,所以這里我再介紹另外一種辦法:重新定義一個(gè)和有參數(shù)的IMP指針相同的指針類型,在獲取IMP時(shí)把它強(qiáng)轉(zhuǎn)為此類型据某。這樣運(yùn)用IMP指針后橡娄,就不需要額外的給ViewController寫新的方法:
還有一個(gè)地方我們需要注意,如果這樣直接調(diào)用IMP的話就會(huì)發(fā)生經(jīng)典的EXC_BAD_ACCESS錯(cuò)誤癣籽,我們定義的IMP指針是一個(gè)有返回值的類型挽唉,而其實(shí)我們獲取的viewDidLoad這個(gè)方法是沒有返回值的滤祖,所以我們需要新定義一個(gè)和IMP相同類型的函數(shù)指針比如VIMP,把他的返回值定位Void瓶籽,這樣如果你修改的方法有返回值就用IMP匠童,沒有返回值就用VIMP。
值得注意的是塑顺,如果你重寫的方法有返回值汤求,不要忘記在最后做return。
總結(jié)
實(shí)際上直接調(diào)用一個(gè)方法的IMP指針的效率是高于調(diào)用方法本身的严拒,所以首昔,如果你有一個(gè)合適的時(shí)機(jī)獲取到方法的IMP的話,你可以試著調(diào)用它糙俗。
這是只是IMP使用的場(chǎng)景之一,它還有許多作用预鬓,希望大家多多發(fā)現(xiàn)巧骚。