方法異常捕獲轉(zhuǎn)發(fā)的流程

引入例子

@interface Person : NSObject

- (void)logString:(NSString *)str;

@end

@implementation Person
@end

Person聲明了一個(gè)方法但是沒有實(shí)現(xiàn)。這時(shí)如果調(diào)用會(huì)崩潰

Person * p = [Person new];    
[p logString:@"哈哈哈哈??"];

消息轉(zhuǎn)發(fā)的過程:

Isa去類對(duì)象中找到方法,然后去發(fā)送消息。找不到的話去父類一層一層往上。如果還沒找到就會(huì)進(jìn)到動(dòng)態(tài)解析。

(注意:isa指向類;類對(duì)象的isa指向元類;如果當(dāng)前類繼承自NSObject,那么元類再指向根源類;如果當(dāng)前類繼承自NSObject,那么元類再指向自己)

動(dòng)態(tài)解析流程:

// 攔截類方法
+ (BOOL)resolveClassMethod:(SEL)sel {
    return NO;
}
// 攔截實(shí)例方法
+ (BOOL)resolveInstanceMethod:(SEL)sel {
    return NO;
}

以此為例,我們要在Person中攔截

@implementation Person

// 注意 `id self,SEL log`前兩個(gè)參數(shù)必須要加上接收,不能只寫str這一個(gè)參數(shù)
void dynamicLogString(id self,SEL log,NSString *str) { // 1.1
    NSLog(@"%@", str);
}

// 攔截實(shí)例方法
+ (BOOL)resolveInstanceMethod:(SEL)sel {
    
    // 動(dòng)態(tài)創(chuàng)建方法接收消息
    if (sel == @selector(logString:)) {
        class_addMethod(self, sel, (IMP)dynamicLogString, @"v@:@"); // 1.2
    }
    return NO;
}

@end
  • 1.1
    我們要引用的動(dòng)態(tài)方法(dynamicLogString為例),必須接收兩個(gè)參數(shù),如果需要接受額外參數(shù),那么必須在后面追加。
  • 1.2
    參數(shù)v@:@是根據(jù)原-logString:方法的配置來定的 v對(duì)應(yīng)void;@對(duì)應(yīng)方法(即id類型);:代表有參數(shù);@代表參數(shù)類型;
    參數(shù)配置連接
    以上就是第一步攔截,但是如果沒有設(shè)置動(dòng)態(tài)方法或者不想通過動(dòng)態(tài)添加方法的方式處理的話,系統(tǒng)就進(jìn)行動(dòng)態(tài)轉(zhuǎn)發(fā);

尋找備用接收者

- (id)forwardingTargetForSelector:(SEL)aSelector
設(shè)置一個(gè)合適的對(duì)象來調(diào)用這個(gè)方法(該對(duì)象聲明+實(shí)現(xiàn)了該方法)

// 攔截實(shí)例方法
+ (BOOL)resolveInstanceMethod:(SEL)sel {
    // 動(dòng)態(tài)創(chuàng)建方法接收消息
//    if (sel == @selector(logString:)) {
//        class_addMethod(self, sel, (IMP)dynamicLogString, @"v@:@");
//    }
    return NO;
}

- (id)forwardingTargetForSelector:(SEL)aSelector {
    if ([TempObjct instancesRespondToSelector:aSelector]) {
        return [TempObjct new];
    }
    return [super forwardingTargetForSelector:aSelector];
}

如果還是沒找到合適的對(duì)象來接收這個(gè)方法的話會(huì)進(jìn)入方法轉(zhuǎn)發(fā)

方法轉(zhuǎn)發(fā)

分兩步:
1.方法簽名
2.方法轉(zhuǎn)發(fā)

- (id)forwardingTargetForSelector:(SEL)aSelector {
//    if ([TempObjct instancesRespondToSelector:aSelector]) {
//        return [TempObjct new];
//    }
    return [super forwardingTargetForSelector:aSelector];
}

// 方法簽名
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector {
    if (aSelector == @selector(logString:)) {
        return [NSMethodSignature signatureWithObjCTypes:"@@:*"];  // 注意這里接受的是c字符串不是@"",不然會(huì)crash
    }
    return [super methodSignatureForSelector:aSelector];
}

// 方法轉(zhuǎn)發(fā)
- (void)forwardInvocation:(NSInvocation *)anInvocation {
    if ([TempObjct instancesRespondToSelector:anInvocation.selector]) {
        [anInvocation invokeWithTarget:[TempObjct new]];
        return;
    }
    // super
    [super forwardInvocation:anInvocation];
}

終極攔截

如果以上方法都沒有攔截到,或者想要保證程序不會(huì)崩潰,可以添加下面方法

- (void)doesNotRecognizeSelector:(SEL)aSelector {
    NSLog(@"未知方法");
}

注意這個(gè)方法會(huì)捕獲到最終所有沒法直行的方法;

以上就是完整流程

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容