日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

詳解iOS Method Swizzling使用陷阱

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

在閱讀團隊一項目源碼時,發現Method Swizzling的寫法有些瑕疵。這篇文章主要就介紹iOS Method Swizzling的正確寫法應該是什么樣的。

下面是iOS Method Swizzling的一種實現:

+ (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);}

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

一個例子

為了說明這個問題,我們先來假設一個場景:

@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的方法也交換成功,但當我們執行[Son1 easeapi]時,發現CRASH了。

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

這就奇怪了,我們HOOK的是Son2的方法,怎么會產生Son1的崩潰?

為什么會發生崩潰

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

首先明確一點,class_getInstanceMethod會查找父類的實現。

在上例中,easeapi是在Son2的父類Father中實現的,執行method_exchangeImplementations之后,Father的easeapi和Son2的new_easeapi進行了方法交換。

交換之后,當Son1(Father的子類)執行easeapi方法時,會通過「消息查找」找到Father的easeapi方法實現。

重點來了!

由于已經發生了方法交換,實際上執行的是Son2的new_easeapi方法。

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

可惡的是,在new_easeapi中執行了[self new_easeapi]。此時這里的self是Son1實例,但Son1及其父類Father中并沒有new_easeapi的SEL,找不到對應的SEL,自然就會CRASH。

什么情況下不會有問題?

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

至少在下面幾種場景中都不會有問題:

Son2中有easeapi的實現

在上例中,如果我們在Son2中重寫了easeapi方法,執行class_getInstanceMethod(class, fromSelector)獲取到的是Son2的easeapi實現,而不是Father的。這樣,執行method_exchangeImplementations后,不會影響到Father的實現。

new_easeapi實現改進

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

在這個場景中,由于不會執行[self new_easeapi],也不會有問題。但這樣就達不到HOOK的效果。

改進優化

推薦的Method Swizzling實現:

+ (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); } });}

可以看到,至少有兩點變化:

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

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

class_addMethod給指定Class添加一個SEL的實現(或者說是SEL和指定IMP的綁定),添加成功返回YES,SEL已經存在或添加失敗返回NO。

它有兩個需要注意的點:

如果該SEL在父類中有實現,則會添加一個覆蓋父類的方法; 如果該Class中已經有SEL,則返回NO。

執行class_addMethod能避免干擾到父類,這也是為什么推薦大家盡量先使用class_addMethod的原因。顯然易見,因為iOS Runtime消息傳遞機制的影響,只執行method_exchangeImplementations操作時可能會影響到父類的方法。基于這個原理,如果HOOK的就是本類中實現的方法,那么直接用method_exchangeImplementations也是完全沒問題的。

class_replaceMethod

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

到此這篇關于詳解iOS Method Swizzling使用陷阱的文章就介紹到這了,更多相關iOS Method Swizzling內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!

標簽: IOS
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
蜜桃av.网站在线观看| 高清av一区| 女人天堂亚洲aⅴ在线观看| 国产精品不卡| 日韩成人a**站| 麻豆mv在线观看| 亚洲天堂1区| 激情久久久久久久| 久久久777| 夜夜嗨av一区二区三区网站四季av| 国产精品av一区二区| 欧美午夜不卡| 亚洲精一区二区三区| 综合欧美亚洲| 国产精品成人自拍| 国内不卡的一区二区三区中文字幕| 久久超级碰碰| 国产不卡精品| 国产91精品对白在线播放| 亚洲视频综合| aa亚洲婷婷| 日韩1区2区3区| 久久超碰99| 97人人精品| 红桃视频国产精品| 日韩精品社区| 国产 日韩 欧美 综合 一区 | 免费看的黄色欧美网站| 亚洲丝袜美腿一区| 国产欧美大片| 欧产日产国产精品视频| aa国产精品| 国产日韩视频| 肉色欧美久久久久久久免费看| 亚洲精品888| 日韩精品a在线观看91| 欧美精品成人| 国精品一区二区三区| 一二三区精品| 久久精品午夜| 亚洲欧美高清| 久久99偷拍| 欧美另类综合| 国产精品一区二区av交换| 91av亚洲| 免费在线观看一区二区三区| 国产日韩亚洲| 亚洲二区精品| 欧美日韩一区二区三区在线电影| 成人在线视频区| 免费在线视频一区| 国产精品99久久免费| 五月天久久久| 国产精品chinese| 欧美日韩国产一区精品一区| 日本a级不卡| 久久国产亚洲| 91精品视频一区二区| 欧美日韩精品免费观看视欧美高清免费大片 | 精品久久久久久久| 国产视频久久| 欧美黄色精品| 男女男精品视频网| 国产资源在线观看入口av| 在线国产精品一区| 欧美一区久久久| 97久久亚洲| 国产综合精品一区| 国产精品亚洲人成在99www| 亚洲韩日在线| 久久99精品久久久久久园产越南| 不卡在线一区二区| 日韩精品视频在线看| 欧美日韩国产v| 欧美日韩中出| 在线一区欧美| 国产精品99一区二区三| 亚洲精品国产日韩| 久久要要av| 精品一区二区三区中文字幕在线| 九色精品91| 国产一区一一区高清不卡| 日韩影院免费视频| 精品亚洲美女网站| 国产亚洲久久| 亚洲自啪免费| 中文另类视频| 青青在线精品| 久久亚洲视频| 一区在线观看| 日韩精品网站| 精品淫伦v久久水蜜桃| 日韩成人av影视| 免费日韩av| 久久精品国产亚洲夜色av网站| 久久影院一区二区三区| 久久精品av麻豆的观看方式| 美美哒免费高清在线观看视频一区二区 | 老鸭窝毛片一区二区三区| 欧美日韩免费看片| 久久精品国产福利| 久久激五月天综合精品| 日韩精品一级二级 | 精品一区欧美| 色88888久久久久久影院| 欧美国产另类| 国产亚洲欧美日韩精品一区二区三区| 亚洲精选久久| 影音先锋久久精品| 丝袜亚洲另类欧美| 亚洲欧美日本视频在线观看| 精品一区免费| 九九综合在线| 亚洲天堂成人| 欧美片第1页| 蜜臀久久精品| 蜜臀国产一区| 亚洲天堂免费电影| 成人免费电影网址| 国产成人精品亚洲日本在线观看| 国产一区二区三区久久久久久久久| 国产乱码精品一区二区三区四区 | 久久av一区| 麻豆亚洲精品| 亚洲日本三级| 日韩欧美中文字幕一区二区三区| 亚洲午夜免费| 日本不卡一区二区| 欧美综合精品| 国产精品亚洲综合久久| 欧美一区二区三区久久| 欧美综合精品| 国产精品观看| 福利一区二区| 日本精品不卡| 欧美高清一区| 在线看片日韩| 日本免费在线视频不卡一不卡二| 亚洲精品国模| 日韩二区在线观看| 国产精品久久久久久妇女| 国产精品免费99久久久| 久久精品一区二区国产| 捆绑调教美女网站视频一区 | 伊人久久高清| 性欧美69xoxoxoxo| 中文字幕中文字幕精品| 欧美日韩精品一区二区三区在线观看| 国产精品二区影院| 欧美日韩视频网站| 黄色欧美日韩| 亚洲区欧美区| 日韩av在线中文字幕| 99精品在线观看| 免费日韩av片| 国产欧美69| 亚洲一区资源| 国产精品呻吟| 91福利精品在线观看| 精品伊人久久久| 欧美综合另类| 青青青国产精品| 久久久久久色| 亚洲欧洲美洲av| 伊人久久亚洲影院| 亚州精品视频| 精品一区二区三区中文字幕视频 | 国产精品yjizz视频网| 蜜臀91精品国产高清在线观看| 综合五月婷婷| 福利精品一区| 蜜桃视频免费观看一区| 国产精品**亚洲精品| 99久久夜色精品国产亚洲狼 | 日韩成人午夜精品| 国产91在线播放精品| 亚洲国产成人精品女人| 午夜久久av| 毛片在线网站| 一区二区三区四区在线观看国产日韩| 麻豆国产精品| 久久国产高清| 精品国产鲁一鲁****| 欧美激情五月| 日韩中文字幕一区二区三区| 免费一级欧美片在线观看网站| 亚洲精品一区三区三区在线观看| 中文在线日韩| 欧洲精品一区二区三区| 午夜在线播放视频欧美| 久久av国产紧身裤| 国产精品外国| 麻豆视频在线看| 日韩精品1区2区3区| 91精品国产成人观看| 欧美视频精品全部免费观看| 99久久久久国产精品| 国产精品免费大片| 免费日本视频一区| 九九精品调教|