iOS 消息轉(zhuǎn)發(fā)機制(實現(xiàn)多繼承)

我們都知道在Objective-C中,對象調(diào)用方法實際上是在發(fā)消息诲泌,當(dāng)對象接收到一條消息時盲赊,消息函數(shù)隨著對象isa指針到類的結(jié)構(gòu)體中,在method list中查找方法selector敷扫。如果在本類中找不到對應(yīng)的selector哀蘑,則objc_msgSend會向其父類的method list中查找selector,如果還不能找到則沿著繼承關(guān)系一直向上查找呻澜,直到找到NSObject類递礼。

如果一直查找到根類仍舊沒有實現(xiàn)惨险,則嘗試進行消息轉(zhuǎn)發(fā)羹幸。消息轉(zhuǎn)發(fā)分為三步:

  • 第一步 動態(tài)方法解析

調(diào)用resolveInstanceMethod:方法 (或 resolveClassMethod:)。允許用戶在此時為該 Class 動態(tài)添加實現(xiàn)辫愉。如果有實現(xiàn)了栅受,則調(diào)用并返回YES,那么重新開始objc_msgSend流程。這一次對象會響應(yīng)這個選擇器屏镊,一般是因為它已經(jīng)調(diào)用過class_addMethod依疼。如果仍沒實現(xiàn),繼續(xù)下面的動作而芥。

  • 第二步 快速轉(zhuǎn)發(fā)(消息轉(zhuǎn)發(fā)重定向)

調(diào)用forwardingTargetForSelector:方法律罢,嘗試找到一個能響應(yīng)該消息的對象。如果獲取到棍丐,則直接把消息轉(zhuǎn)發(fā)給它误辑,返回非 nil 對象。否則返回 nil 歌逢,繼續(xù)下面的動作巾钉。注意,這里不要返回 self 秘案,否則會形成死循環(huán)砰苍。

  • 第三步 完整消息轉(zhuǎn)發(fā)
    • 調(diào)用methodSignatureForSelector:方法,嘗試獲得一個方法簽名阱高。如果獲取不到赚导,則直接調(diào)用doesNotRecognizeSelector拋出異常。如果能獲取赤惊,則返回非nil創(chuàng)建一個 NSlnvocation 并傳給forwardInvocation:辟癌。

    • 調(diào)用forwardInvocation:方法,將第3步獲取到的方法簽名包裝成 Invocation 傳入荐捻,如何處理就在這里面了黍少,并返回非nil

  • 最后

調(diào)用doesNotRecognizeSelector:处面,默認(rèn)的實現(xiàn)是拋出異常厂置。如果第3步?jīng)]能獲得一個方法簽名,執(zhí)行該步驟魂角。


消息轉(zhuǎn)發(fā)的實際應(yīng)用

第一步 動態(tài)方法解析
// 消息轉(zhuǎn)發(fā)第一步 動態(tài)方法解析(其實不是轉(zhuǎn)發(fā)昵济,是動態(tài)添加方法)
+ (BOOL)resolveInstanceMethod:(SEL)sel{
    NSString *selName = NSStringFromSelector(sel);

     // 此處可根據(jù)情況作出需要的判斷,需要給哪些方法做處理
    if ([selName hasPrefix:@"eat"]) {
        class_addMethod([self class], sel, (IMP)shouldDoSomeThing, "v@:");
        return YES;
    }
    return [super resolveInstanceMethod:sel];
}

void shouldDoSomeThing(id self ,SEL _cmd) {
    NSLog(@"class:%@, sel:%s",self,sel_getName(_cmd));
    NSLog(@"差不多也該干點啥");
}

如果沒有實現(xiàn)第一步野揪,系統(tǒng)會繼續(xù)進行第二步查找访忿,進行快速轉(zhuǎn)發(fā)

第二步 快速轉(zhuǎn)發(fā)(消息轉(zhuǎn)發(fā)重定向)
// 消息轉(zhuǎn)發(fā)第二步 轉(zhuǎn)發(fā)給備選接收者
- (id)forwardingTargetForSelector:(SEL)aSelector{
    if ([HitTestView instancesRespondToSelector:aSelector]) {
        return [HitTestView new];
    }
    
    return [super forwardingTargetForSelector:aSelector];
}

如果沒有實現(xiàn)第二步,系統(tǒng)會繼續(xù)進行第三步查找斯稳,進行完整消息轉(zhuǎn)發(fā)

  • 第三步 完整消息轉(zhuǎn)發(fā)
//消息轉(zhuǎn)發(fā)第三步 完整的消息轉(zhuǎn)發(fā)
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector{
    NSMethodSignature *signature = [super methodSignatureForSelector:aSelector];
    if (!signature) {
        //不簽名會崩潰海铆,以下兩種獲取方法簽名的方式皆可

        // 1. 判斷實現(xiàn)類的實例是否有這個方法 有則簽名這個方法 保證能正確轉(zhuǎn)發(fā)
//        if ([TestObject instancesRespondToSelector:aSelector]) {
//            signature = [TestObject instanceMethodSignatureForSelector:aSelector];
//        }
        // 2. 直接簽名
        if ([NSStringFromSelector(aSelector) hasPrefix:@"eat"]) {
            signature = [NSMethodSignature signatureWithObjCTypes:"v@:"];
        }
    }
    
    return signature;
}

- (void)forwardInvocation:(NSInvocation *)anInvocation{
    if ([TestObject instancesRespondToSelector:anInvocation.selector]) {
        [anInvocation invokeWithTarget:[TestObject new]];
    }
}

至此,若沒有實現(xiàn)以上三步挣惰,調(diào)用doesNotRecognizeSelector:卧斟,拋出異常殴边。

注:類型編碼:"v@:"
v -> void 表示無返回值
@ -> object 表示id參數(shù)(self)
: -> method selector 表示SEL
"v@:@":表示傳入一個參數(shù),無返回值


消息轉(zhuǎn)發(fā)+Protocol實現(xiàn)多繼承

現(xiàn)有 MultiInheritA 珍语、 MultiInheritB 锤岸、 MultiInheritC 這三個類,讓C繼承A也繼承B的方法板乙。

先看 MultiInheritA

#import <Foundation/Foundation.h>

@protocol MultiInheritAProtocol <NSObject>

- (void)goToSchool;

@end

@interface MultiInheritA : NSObject<MultiInheritAProtocol>

@end

#import "MultiInheritA.h"

@implementation MultiInheritA

- (void)goToSchool {
    NSLog(@"MultiInheritA goToSchool");
}

@end



再看 MultiInheritB

#import <Foundation/Foundation.h>

@protocol MultiInheritBProtocol <NSObject>

- (void)goHome;

@end

@interface MultiInheritB : NSObject <MultiInheritBProtocol>

@end
#import "MultiInheritB.h"

@implementation MultiInheritB

- (void)goHome {
    NSLog(@"MultiInheritB goHome");
}

@end


重點是這個 MultiInheritC

#import <Foundation/Foundation.h>
#import "MultiInheritA.h"
#import "MultiInheritB.h"

@interface MultiInheritC : NSObject<MultiInheritAProtocol,MultiInheritBProtocol>

@end
#import "MultiInheritC.h"


@interface MultiInheritC()

@property (nonatomic,strong) MultiInheritA *myA;
@property (nonatomic,strong) MultiInheritB *myB;

@end

@implementation MultiInheritC

- (MultiInheritA *)myA {
    if (_myA == nil) {
        _myA = [MultiInheritA new];
    }
    return _myA;
}

- (MultiInheritB *)myB {
    if (_myB == nil) {
        _myB = [MultiInheritB new];
    }
    return _myB;
}

- (id)forwardingTargetForSelector:(SEL)aSelector {
    if ([self.myA respondsToSelector:aSelector]) {
        return self.myA;
    }
    else if ([self.myB respondsToSelector:aSelector]) {
        return self.myB;
    }
    return self;
}

@end
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末是偷,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子募逞,更是在濱河造成了極大的恐慌晓猛,老刑警劉巖,帶你破解...
    沈念sama閱讀 211,123評論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件凡辱,死亡現(xiàn)場離奇詭異戒职,居然都是意外死亡,警方通過查閱死者的電腦和手機透乾,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,031評論 2 384
  • 文/潘曉璐 我一進店門洪燥,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人乳乌,你說我怎么就攤上這事捧韵。” “怎么了汉操?”我有些...
    開封第一講書人閱讀 156,723評論 0 345
  • 文/不壞的土叔 我叫張陵再来,是天一觀的道長。 經(jīng)常有香客問我磷瘤,道長芒篷,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,357評論 1 283
  • 正文 為了忘掉前任采缚,我火速辦了婚禮针炉,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘扳抽。我一直安慰自己篡帕,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 65,412評論 5 384
  • 文/花漫 我一把揭開白布贸呢。 她就那樣靜靜地躺著镰烧,像睡著了一般。 火紅的嫁衣襯著肌膚如雪楞陷。 梳的紋絲不亂的頭發(fā)上怔鳖,一...
    開封第一講書人閱讀 49,760評論 1 289
  • 那天,我揣著相機與錄音猜谚,去河邊找鬼败砂。 笑死赌渣,一個胖子當(dāng)著我的面吹牛魏铅,可吹牛的內(nèi)容都是我干的昌犹。 我是一名探鬼主播,決...
    沈念sama閱讀 38,904評論 3 405
  • 文/蒼蘭香墨 我猛地睜開眼览芳,長吁一口氣:“原來是場噩夢啊……” “哼斜姥!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起沧竟,我...
    開封第一講書人閱讀 37,672評論 0 266
  • 序言:老撾萬榮一對情侶失蹤铸敏,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后悟泵,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體杈笔,經(jīng)...
    沈念sama閱讀 44,118評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,456評論 2 325
  • 正文 我和宋清朗相戀三年糕非,在試婚紗的時候發(fā)現(xiàn)自己被綠了蒙具。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 38,599評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡朽肥,死狀恐怖禁筏,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情衡招,我是刑警寧澤篱昔,帶...
    沈念sama閱讀 34,264評論 4 328
  • 正文 年R本政府宣布,位于F島的核電站始腾,受9級特大地震影響州刽,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜浪箭,卻給世界環(huán)境...
    茶點故事閱讀 39,857評論 3 312
  • 文/蒙蒙 一怀伦、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧山林,春花似錦房待、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,731評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至框冀,卻和暖如春流椒,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背明也。 一陣腳步聲響...
    開封第一講書人閱讀 31,956評論 1 264
  • 我被黑心中介騙來泰國打工宣虾, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留惯裕,地道東北人。 一個月前我還...
    沈念sama閱讀 46,286評論 2 360
  • 正文 我出身青樓绣硝,卻偏偏與公主長得像蜻势,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子鹉胖,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 43,465評論 2 348

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