Objective-C Runtime 运行时之四:Method Swizzling
理解Method Swizzling是學(xué)習(xí)runtime機制的一個很好的機會。在此不多做整理,僅翻譯由Mattt Thompson發(fā)表于nshipster的Method Swizzling一文。
Method Swizzling是改變一個selector的實際實現(xiàn)的技術(shù)。通過這一技術(shù),我們可以在運行時通過修改類的分發(fā)表中selector對應(yīng)的函數(shù),來修改方法的實現(xiàn)。
例如,我們想跟蹤在程序中每一個view controller展示給用戶的次數(shù):當然,我們可以在每個view controller的viewDidAppear中添加跟蹤代碼;但是這太過麻煩,需要在每個view controller中寫重復(fù)的代碼。創(chuàng)建一個子類可能是一種實現(xiàn)方式,但需要同時創(chuàng)建UIViewController,?UITableViewController,?UINavigationController及其它UIKit中view controller的子類,這同樣會產(chǎn)生許多重復(fù)的代碼。
這種情況下,我們就可以使用Method Swizzling,如在代碼所示:
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 | #import <objc/runtime.h> @implementation UIViewController (Tracking) + (void)load { ????????static dispatch_once_t onceToken; ????dispatch_once(&onceToken, ^{ ????????Class class = [self class];???????? ????????// When swizzling a class method, use the following: ????????????????????// Class class = object_getClass((id)self); ????????SEL originalSelector = @selector(viewWillAppear:); ????????????????????SEL swizzledSelector = @selector(xxx_viewWillAppear:); ????????Method originalMethod = class_getInstanceMethod(class, originalSelector); ????????????????????Method swizzledMethod = class_getInstanceMethod(class, swizzledSelector); ????????BOOL didAddMethod = ????????????????????????class_addMethod(class, ????????????????originalSelector, ????????????????method_getImplementation(swizzledMethod), ????????????????method_getTypeEncoding(swizzledMethod)); ????????if (didAddMethod) { ????????????????????????class_replaceMethod(class, ????????????????swizzledSelector, ????????????????method_getImplementation(originalMethod), ????????????????method_getTypeEncoding(originalMethod)); ????????} else { ????????????method_exchangeImplementations(originalMethod, swizzledMethod); ????????} ????}); } #pragma mark - Method Swizzling - (void)xxx_viewWillAppear:(BOOL)animated { ????????[self xxx_viewWillAppear:animated]; ????NSLog(@"viewWillAppear: %@", self); } @end |
在這里,我們通過method swizzling修改了UIViewController的@selector(viewWillAppear:)對應(yīng)的函數(shù)指針,使其實現(xiàn)指向了我們自定義的xxx_viewWillAppear的實現(xiàn)。這樣,當UIViewController及其子類的對象調(diào)用viewWillAppear時,都會打印一條日志信息。
上面的例子很好地展示了使用method swizzling來一個類中注入一些我們新的操作。當然,還有許多場景可以使用method swizzling,在此不多舉例。在此我們說說使用method swizzling需要注意的一些問題:
Swizzling應(yīng)該總是在+load中執(zhí)行
在Objective-C中,運行時會自動調(diào)用每個類的兩個方法。+load會在類初始加載時調(diào)用,+initialize會在第一次調(diào)用類的類方法或?qū)嵗椒ㄖ氨徽{(diào)用。這兩個方法是可選的,且只有在實現(xiàn)了它們時才會被調(diào)用。由于method swizzling會影響到類的全局狀態(tài),因此要盡量避免在并發(fā)處理中出現(xiàn)競爭的情況。+load能保證在類的初始化過程中被加載,并保證這種改變應(yīng)用級別的行為的一致性。相比之下,+initialize在其執(zhí)行時不提供這種保證—事實上,如果在應(yīng)用中沒為給這個類發(fā)送消息,則它可能永遠不會被調(diào)用。
Swizzling應(yīng)該總是在dispatch_once中執(zhí)行
與上面相同,因為swizzling會改變?nèi)譅顟B(tài),所以我們需要在運行時采取一些預(yù)防措施。原子性就是這樣一種措施,它確保代碼只被執(zhí)行一次,不管有多少個線程。GCD的dispatch_once可以確保這種行為,我們應(yīng)該將其作為method swizzling的最佳實踐。
選擇器、方法與實現(xiàn)
在Objective-C中,選擇器(selector)、方法(method)和實現(xiàn)(implementation)是運行時中一個特殊點,雖然在一般情況下,這些術(shù)語更多的是用在消息發(fā)送的過程描述中。
以下是Objective-C Runtime Reference中的對這幾個術(shù)語一些描述:
理解這幾個術(shù)語之間的關(guān)系最好的方式是:一個類維護一個運行時可接收的消息分發(fā)表;分發(fā)表中的每個入口是一個方法(Method),其中key是一個特定名稱,即選擇器(SEL),其對應(yīng)一個實現(xiàn)(IMP),即指向底層C函數(shù)的指針。
為了swizzle一個方法,我們可以在分發(fā)表中將一個方法的現(xiàn)有的選擇器映射到不同的實現(xiàn),而將該選擇器對應(yīng)的原始實現(xiàn)關(guān)聯(lián)到一個新的選擇器中。
調(diào)用_cmd
我們回過頭來看看前面新的方法的實現(xiàn)代碼:
| 1 2 3 4 | - (void)xxx_viewWillAppear:(BOOL)animated { ????[self xxx_viewWillAppear:animated]; ????NSLog(@"viewWillAppear: %@", NSStringFromClass([self class])); } |
咋看上去是會導(dǎo)致無限循環(huán)的。但令人驚奇的是,并沒有出現(xiàn)這種情況。在swizzling的過程中,方法中的[self xxx_viewWillAppear:animated]已經(jīng)被重新指定到UIViewController類的-viewWillAppear:中。在這種情況下,不會產(chǎn)生無限循環(huán)。不過如果我們調(diào)用的是[self viewWillAppear:animated],則會產(chǎn)生無限循環(huán),因為這個方法的實現(xiàn)在運行時已經(jīng)被重新指定為xxx_viewWillAppear:了。
注意事項
Swizzling通常被稱作是一種黑魔法,容易產(chǎn)生不可預(yù)知的行為和無法預(yù)見的后果。雖然它不是最安全的,但如果遵從以下幾點預(yù)防措施的話,還是比較安全的:
轉(zhuǎn)載于:https://www.cnblogs.com/yjg2014/p/5857647.html
總結(jié)
以上是生活随笔為你收集整理的Objective-C Runtime 运行时之四:Method Swizzling的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: Python爬虫实战(5):模拟登录淘宝
- 下一篇: go read text file in