方法异常捕获转发的流程

2020-01-10  本文已影响0人  zolobdz

引入例子

@interface Person : NSObject

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

@end

@implementation Person
@end

Person声明了一个方法但是没有实现。这时如果调用会崩溃

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

消息转发的过程:

Isa去类对象中找到方法,然后去发送消息。找不到的话去父类一层一层往上。如果还没找到就会进到动态解析。

(注意:isa指向类;类对象的isa指向元类;如果当前类继承自NSObject,那么元类再指向根源类;如果当前类继承自NSObject,那么元类再指向自己)

动态解析流程:

// 拦截类方法
+ (BOOL)resolveClassMethod:(SEL)sel {
    return NO;
}
// 拦截实例方法
+ (BOOL)resolveInstanceMethod:(SEL)sel {
    return NO;
}

以此为例,我们要在Person中拦截

@implementation Person

// 注意 `id self,SEL log`前两个参数必须要加上接收,不能只写str这一个参数
void dynamicLogString(id self,SEL log,NSString *str) { // 1.1
    NSLog(@"%@", str);
}

// 拦截实例方法
+ (BOOL)resolveInstanceMethod:(SEL)sel {
    
    // 动态创建方法接收消息
    if (sel == @selector(logString:)) {
        class_addMethod(self, sel, (IMP)dynamicLogString, @"v@:@"); // 1.2
    }
    return NO;
}

@end

寻找备用接收者

- (id)forwardingTargetForSelector:(SEL)aSelector
设置一个合适的对象来调用这个方法(该对象声明+实现了该方法)

// 拦截实例方法
+ (BOOL)resolveInstanceMethod:(SEL)sel {
    // 动态创建方法接收消息
//    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];
}

如果还是没找到合适的对象来接收这个方法的话会进入方法转发

方法转发

分两步:
1.方法签名
2.方法转发

- (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字符串不是@"",不然会crash
    }
    return [super methodSignatureForSelector:aSelector];
}

// 方法转发
- (void)forwardInvocation:(NSInvocation *)anInvocation {
    if ([TempObjct instancesRespondToSelector:anInvocation.selector]) {
        [anInvocation invokeWithTarget:[TempObjct new]];
        return;
    }
    // super
    [super forwardInvocation:anInvocation];
}

终极拦截

如果以上方法都没有拦截到,或者想要保证程序不会崩溃,可以添加下面方法

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

注意这个方法会捕获到最终所有没法直行的方法;

以上就是完整流程

上一篇下一篇

猜你喜欢

热点阅读