Method Swizzling

前言

前段時(shí)間去面試的時(shí)候有一題問(wèn)的是method swizzling是什么猾普?請(qǐng)簡(jiǎn)述原理以及如何使用虾啦?oc的runtime確實(shí)是沒(méi)研究過(guò),不過(guò)既然遇到了锭吨,還是順手整理下來(lái)。

簡(jiǎn)介

在OC的runtime中寒匙,當(dāng)一個(gè)消息被發(fā)送給一個(gè)對(duì)象零如,就有一個(gè)方法會(huì)被調(diào)用,這意味著一個(gè)給定的Selector對(duì)應(yīng)的method可以在runtime中發(fā)生改變锄弱。利用這一特性可以在不知道一個(gè)類的源碼以及實(shí)現(xiàn)原理的情況下考蕾,不需要通過(guò)繼承或者重寫來(lái)改變這個(gè)類的功能。與繼承和重寫相比会宪,被改變的功能適用于這個(gè)類以及它的子類肖卧。這個(gè)黑魔法就叫method swizzling

實(shí)現(xiàn)原理

Method Swizzling是改變一個(gè)Selector的實(shí)際實(shí)現(xiàn)的技術(shù)掸鹅。通過(guò)這一技術(shù)塞帐,我們可以在運(yùn)行時(shí)通過(guò)修改類的分發(fā)表中selector對(duì)應(yīng)的函數(shù),來(lái)修改方法的實(shí)現(xiàn)巍沙。

一個(gè)類中的方法列表包含了一系列的Selector映射列表葵姥,當(dāng)給定一個(gè)method時(shí),動(dòng)態(tài)消息系統(tǒng)會(huì)在這個(gè)列表中尋找對(duì)應(yīng)的實(shí)現(xiàn)方法句携。這些實(shí)現(xiàn)方法被存儲(chǔ)在一個(gè)叫做IMP的方法指針中榔幸,它有一個(gè)屬性:id (IMP *)(id, SEL, ...)

這里可能有點(diǎn)繞,method矮嫉,Selector削咆,message總覺(jué)得差不多。這里需要了解下這幾個(gè)的概念蠢笋,否則下面講的可能就會(huì)迷迷糊糊的了拨齐。

  • Selector

a Selector is the name of a method.

Selector是一個(gè)方法的名稱,例如咱們都非常熟悉的alloc, init, release, dictionaryWithObjectsAndKeys:, setObject:forKey:在開發(fā)過(guò)程中挺尿,指定按鈕的點(diǎn)擊事件時(shí)常用到的@selector(doSomething:),就是指定了方法名奏黑。

  • message:

a message is a selector and the arguments you are sending with it.

message就是包含有參數(shù)的Selector炊邦,例如[dictionary setObject:obj forKey:key],;這里的Selector就是setObject:forKey:

  • method

a method is a combination of a selector and an implementation (and accompanying metadata).

method是Selector和implementation的結(jié)合

還有:

  • implementation

the actual executable code of a method. Its type at runtime is an IMP, and it's really just a function pointer.

好吧熟史,這個(gè)概念應(yīng)該比較不會(huì)混淆馁害,不過(guò)需要注意的是,implementation在runtime中就是一個(gè)函數(shù)指針

一個(gè)類維護(hù)一個(gè)運(yùn)行時(shí)可接收的消息分發(fā)表蹂匹;分發(fā)表中的每個(gè)入口是一個(gè)方法(Method)碘菜,其中key是一個(gè)特定名稱,即選擇器(SEL)限寞,其對(duì)應(yīng)一個(gè)實(shí)現(xiàn)(IMP)忍啸,即指向底層C函數(shù)的指針。

簡(jiǎn)單使用

在NSString中履植,有三個(gè)selector:lowercaseString, uppercaseString, capitalizedString分別對(duì)應(yīng)三個(gè)不同的IMP:

NSString's selector

使用method swizzling可以在這個(gè)列表中添加一個(gè)新的selector计雌,或者交換兩個(gè)selector對(duì)應(yīng)的指針,之后這個(gè)表格是長(zhǎng)這樣的:

增加交換指針
交換兩個(gè)selector對(duì)應(yīng)的指針

交換兩個(gè)指針需要用到的方法有:

//交換兩個(gè)method的對(duì)應(yīng)的implementation玫霎,也就是IMP
void method_exchangeImplementation(Method m1, Method m2)

//獲取一個(gè)類中給定的selector對(duì)應(yīng)的method
Method class_getInstanceMethod(Class aClass, SEL aSelector)

使用這兩個(gè)函數(shù)就能交換對(duì)應(yīng)的指針了:

//獲取兩個(gè)selector對(duì)應(yīng)的method
Method originalMehtod = class_getInstanceMthod([NSString class],@selector(lowercaseString) );
Method swappedMethod = class_getInstanceMethod([NSString class], @selector(uppercaseString));

//交換兩個(gè)method
god method_exchangeImplementation(originalMethod,swappedMethod);

這么做之后凿滤,當(dāng)你調(diào)用NSString的lowercaseString方法執(zhí)行的功能會(huì)變成uppercaseString所實(shí)現(xiàn)的功能。

添加一個(gè)selector

在實(shí)際開發(fā)過(guò)程中庶近,交換兩個(gè)方法的功能用處不大翁脆,但是添加一個(gè)功能就不同了。
例如現(xiàn)在想要在每次調(diào)用lowercaseString方法的時(shí)候加一個(gè)log鼻种,說(shuō)明我調(diào)用了這個(gè)方法反番,可以在寫一個(gè)NSString的category,并在這個(gè)類中添加這個(gè)功能:

@interface NSString (NSStringLowerCaseLog)
- (NSString *)myLowerCaseString;
@end


@implementation NSString (NSStringLowerCaseLog)
+ (void)load {
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        Method originalMethod = class_getInstanceMethod([NSString class], @selector(lowercaseString));
        Method swapteMethod = class_getInstanceMethod([NSString class], @selector(myLowercaseString));
        method_exchangeImplementations(originalMethod, swapteMethod);
    });
}

- (NSString *)myLowercaseString {
    //相當(dāng)于調(diào)用了之前的'lowercaseString'方法
    NSString *lowercase = [self myLowercaseString];
    NSLog(@"%@ => %@",self, lowercase);
    return lowercase;
}
@end

注意事項(xiàng):

1. Swizzling應(yīng)該總是在+load中執(zhí)行

在Objective-C中叉钥,運(yùn)行時(shí)會(huì)自動(dòng)調(diào)用每個(gè)類的兩個(gè)方法罢缸。+load會(huì)在類初始加載時(shí)調(diào)用,+initialize會(huì)在第一次調(diào)用類的類方法或?qū)嵗椒ㄖ氨徽{(diào)用投队。這兩個(gè)方法是可選的祖能,且只有在實(shí)現(xiàn)了它們時(shí)才會(huì)被調(diào)用。由于method swizzling會(huì)影響到類的全局狀態(tài)蛾洛,因此要盡量避免在并發(fā)處理中出現(xiàn)競(jìng)爭(zhēng)的情況养铸。+load能保證在類的初始化過(guò)程中被加載,并保證這種改變應(yīng)用級(jí)別的行為的一致性轧膘。相比之下钞螟,+initialize在其執(zhí)行時(shí)不提供這種保證—事實(shí)上,如果在應(yīng)用中沒(méi)為給這個(gè)類發(fā)送消息谎碍,則它可能永遠(yuǎn)不會(huì)被調(diào)用鳞滨。

2. Swizzling應(yīng)該總是在dispatch_once中執(zhí)行

與上面相同,因?yàn)閟wizzling會(huì)改變?nèi)譅顟B(tài)蟆淀,所以我們需要在運(yùn)行時(shí)采取一些預(yù)防措施拯啦。原子性就是這樣一種措施澡匪,它確保代碼只被執(zhí)行一次,不管有多少個(gè)線程褒链。GCD的dispatch_once可以確保這種行為唁情,我們應(yīng)該將其作為method swizzling的最佳實(shí)踐。

參考

Objective-C Runtime 運(yùn)行時(shí)之四:Method Swizzling
Method Swizzling
stackoverflow: What's the difference between a method and a selector?

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末甫匹,一起剝皮案震驚了整個(gè)濱河市甸鸟,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌兵迅,老刑警劉巖抢韭,帶你破解...
    沈念sama閱讀 218,036評(píng)論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異恍箭,居然都是意外死亡刻恭,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,046評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門扯夭,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)吠各,“玉大人,你說(shuō)我怎么就攤上這事勉抓。” “怎么了候学?”我有些...
    開封第一講書人閱讀 164,411評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵藕筋,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我梳码,道長(zhǎng)隐圾,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,622評(píng)論 1 293
  • 正文 為了忘掉前任掰茶,我火速辦了婚禮暇藏,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘濒蒋。我一直安慰自己盐碱,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,661評(píng)論 6 392
  • 文/花漫 我一把揭開白布沪伙。 她就那樣靜靜地躺著瓮顽,像睡著了一般。 火紅的嫁衣襯著肌膚如雪围橡。 梳的紋絲不亂的頭發(fā)上暖混,一...
    開封第一講書人閱讀 51,521評(píng)論 1 304
  • 那天,我揣著相機(jī)與錄音翁授,去河邊找鬼拣播。 笑死晾咪,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的贮配。 我是一名探鬼主播谍倦,決...
    沈念sama閱讀 40,288評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼源葫,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼煌茴!你這毒婦竟也來(lái)了霎俩?” 一聲冷哼從身側(cè)響起坏为,我...
    開封第一講書人閱讀 39,200評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤历葛,失蹤者是張志新(化名)和其女友劉穎锭沟,沒(méi)想到半個(gè)月后致燥,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體蹭沛,經(jīng)...
    沈念sama閱讀 45,644評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡辽剧,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,837評(píng)論 3 336
  • 正文 我和宋清朗相戀三年送淆,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片怕轿。...
    茶點(diǎn)故事閱讀 39,953評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡偷崩,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出撞羽,到底是詐尸還是另有隱情阐斜,我是刑警寧澤,帶...
    沈念sama閱讀 35,673評(píng)論 5 346
  • 正文 年R本政府宣布诀紊,位于F島的核電站谒出,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏邻奠。R本人自食惡果不足惜笤喳,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,281評(píng)論 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望碌宴。 院中可真熱鬧杀狡,春花似錦、人聲如沸贰镣。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,889評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)碑隆。三九已至董朝,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間干跛,已是汗流浹背子姜。 一陣腳步聲響...
    開封第一講書人閱讀 33,011評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人哥捕。 一個(gè)月前我還...
    沈念sama閱讀 48,119評(píng)論 3 370
  • 正文 我出身青樓牧抽,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親遥赚。 傳聞我的和親對(duì)象是個(gè)殘疾皇子扬舒,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,901評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容