亚洲精品久久久中文字幕-亚洲精品久久片久久-亚洲精品久久青草-亚洲精品久久婷婷爱久久婷婷-亚洲精品久久午夜香蕉

您的位置:首頁技術(shù)文章
文章詳情頁

詳解iOS Method Swizzling使用陷阱

瀏覽:2日期:2022-09-17 08:18:39

在閱讀團(tuán)隊(duì)一項(xiàng)目源碼時(shí),發(fā)現(xiàn)Method Swizzling的寫法有些瑕疵。這篇文章主要就介紹iOS Method Swizzling的正確寫法應(yīng)該是什么樣的。

下面是iOS Method Swizzling的一種實(shí)現(xiàn):

+ (void)load { Class class = [self class]; SEL fromSelector = @selector(func); SEL toSelector = @selector(easeapi_func); Method fromMethod = class_getInstanceMethod(class, fromSelector); Method toMethod = class_getInstanceMethod(class, toSelector); method_exchangeImplementations(fromMethod, toMethod);}

這種寫法在一些時(shí)候能正常工作,但實(shí)際上有些問題。那么問題在哪里呢?

一個(gè)例子

為了說明這個(gè)問題,我們先來假設(shè)一個(gè)場(chǎng)景:

@interface Father: NSObject-(void)easeapi;@end@implementation Father-(void)easeapi { //your code}@end//Son1繼承自Father@interface Son1: Father@end@implementation Son1@end//Son2繼承自Father,并HOOK了easeapi方法。@interface Son2: Father@end@implementation Son2+ (void)load { Class class = [self class]; SEL fromSelector = @selector(easeapi); SEL toSelector = @selector(new_easeapi); Method fromMethod = class_getInstanceMethod(class, fromSelector); Method toMethod = class_getInstanceMethod(class, toSelector); method_exchangeImplementations(fromMethod, toMethod);}-(void)new_easeapi { [self new_easeapi]; //your code}@end

看樣子沒什么問題,Son2的方法也交換成功,但當(dāng)我們執(zhí)行[Son1 easeapi]時(shí),發(fā)現(xiàn)CRASH了。

’-[Son1 new_easeapi]: unrecognized selector sent to instance 0x600002d701f0’’

這就奇怪了,我們HOOK的是Son2的方法,怎么會(huì)產(chǎn)生Son1的崩潰?

為什么會(huì)發(fā)生崩潰

要解釋這個(gè)問題,還是要回到原理上。

首先明確一點(diǎn),class_getInstanceMethod會(huì)查找父類的實(shí)現(xiàn)。

在上例中,easeapi是在Son2的父類Father中實(shí)現(xiàn)的,執(zhí)行method_exchangeImplementations之后,F(xiàn)ather的easeapi和Son2的new_easeapi進(jìn)行了方法交換。

交換之后,當(dāng)Son1(Father的子類)執(zhí)行easeapi方法時(shí),會(huì)通過「消息查找」找到Father的easeapi方法實(shí)現(xiàn)。

重點(diǎn)來了!

由于已經(jīng)發(fā)生了方法交換,實(shí)際上執(zhí)行的是Son2的new_easeapi方法。

-(void)new_easeapi { [self new_easeapi]; //your code}

可惡的是,在new_easeapi中執(zhí)行了[self new_easeapi]。此時(shí)這里的self是Son1實(shí)例,但Son1及其父類Father中并沒有new_easeapi的SEL,找不到對(duì)應(yīng)的SEL,自然就會(huì)CRASH。

什么情況下不會(huì)有問題?

上面說了:「這種寫法在一些時(shí)候能正常工作」。那么,到底什么時(shí)候直接執(zhí)行method_exchangeImplementations不會(huì)有問題呢?

至少在下面幾種場(chǎng)景中都不會(huì)有問題:

Son2中有easeapi的實(shí)現(xiàn)

在上例中,如果我們?cè)赟on2中重寫了easeapi方法,執(zhí)行class_getInstanceMethod(class, fromSelector)獲取到的是Son2的easeapi實(shí)現(xiàn),而不是Father的。這樣,執(zhí)行method_exchangeImplementations后,不會(huì)影響到Father的實(shí)現(xiàn)。

new_easeapi實(shí)現(xiàn)改進(jìn)

- (void) new_easeapi { //[self new_easeapi];//屏蔽掉這句代碼 //your code}

在這個(gè)場(chǎng)景中,由于不會(huì)執(zhí)行[self new_easeapi],也不會(huì)有問題。但這樣就達(dá)不到HOOK的效果。

改進(jìn)優(yōu)化

推薦的Method Swizzling實(shí)現(xiàn):

+ (void)load { static dispatch_once_t onceToken; dispatch_once(&onceToken, ^{ Class class = [self class]; SEL fromSelector = @selector(easeapi); SEL toSelector = @selector(new_easeapi); Method fromMethod = class_getInstanceMethod(class, fromSelector); Method toMethod = class_getInstanceMethod(class, toSelector); if(class_addMethod(class, fromSelector, method_getImplementation(toMethod), method_getTypeEncoding(toMethod))) { class_replaceMethod(class, toSelector, method_getImplementation(fromMethod), method_getTypeEncoding(fromMethod)); } else { method_exchangeImplementations(fromMethod, toMethod); } });}

可以看到,至少有兩點(diǎn)變化:

dispatch_once盡管dyld能夠保證調(diào)用Class的load時(shí)是線程安全的,但還是推薦使用dispatch_once做保護(hù),防止極端情況下load被顯示強(qiáng)制調(diào)用時(shí),重復(fù)交換(第一次交換成功,下次又換回來了...),造成邏輯混亂。

增加了class_addMethod判斷class_addMethod & class_replaceMethod還是從定義上理解。

class_addMethod給指定Class添加一個(gè)SEL的實(shí)現(xiàn)(或者說是SEL和指定IMP的綁定),添加成功返回YES,SEL已經(jīng)存在或添加失敗返回NO。

它有兩個(gè)需要注意的點(diǎn):

如果該SEL在父類中有實(shí)現(xiàn),則會(huì)添加一個(gè)覆蓋父類的方法; 如果該Class中已經(jīng)有SEL,則返回NO。

執(zhí)行class_addMethod能避免干擾到父類,這也是為什么推薦大家盡量先使用class_addMethod的原因。顯然易見,因?yàn)閕OS Runtime消息傳遞機(jī)制的影響,只執(zhí)行method_exchangeImplementations操作時(shí)可能會(huì)影響到父類的方法?;谶@個(gè)原理,如果HOOK的就是本類中實(shí)現(xiàn)的方法,那么直接用method_exchangeImplementations也是完全沒問題的。

class_replaceMethod

如果該Class不存在指定SEL,則class_replaceMethod的作用就和class_addMethod一樣; 如果該Class存在指定的SEL,則class_replaceMethod的作用就和method_setImplementation一樣。

到此這篇關(guān)于詳解iOS Method Swizzling使用陷阱的文章就介紹到這了,更多相關(guān)iOS Method Swizzling內(nèi)容請(qǐng)搜索好吧啦網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持好吧啦網(wǎng)!

標(biāo)簽: IOS
相關(guān)文章:
主站蜘蛛池模板: 成人aaaa | 最新亚洲| 日本 3344www高清在线 | 成人做爰全过程免费看网站 | 午夜男人女人爽爽爽视频 | 国产大秀视频在线一区二区 | 欧美中文字幕一区二区三区 | 综合五月婷婷 | 亚洲成在人线久久综合 | 成人免费播放视频777777 | 中文字幕一区二区三区精彩视频 | 亚洲黄色免费在线观看 | 97超视频在线观看 | 欧美激情在线播放第16页 | 成人午夜视频免费 | 亚洲第一黄色网址 | 农村女人的一级毛片 | 麻豆视频网站在线观看 | 亚洲欧美在线精品 | 亚洲成a人片毛片在线 | 国产在线a不卡免费视频 | 亚洲精品久久久久久中文字幕小说 | 国产nv精品你懂得 | 国产午夜精品一区二区三区嫩草 | 国产小视频免费 | 国产淫语对白在线视频 | 国产精品嫩草影院在线 | 九九夜夜| 日韩在线播放视频 | 33333在线亚洲| 欧美日韩国产在线播放 | 国产啪视频1000部免费视频 | 欧美一级特黄高清免费 | 久久我们这里只有精品国产4 | 经典三级影院 | 黄色大片一级片 | 99在线精品视频免费观里 | 日本精品久久久久久久 | 毛片大片免费看 | 成 人 黄 色 大 片全部 | 国产精品精品国产一区二区 |