iOS--消息转发机制

iOS–消息转发机制

相信大家对这句话unrecognized selector sent to instance 0x*********一点都不陌生吧。

下面就来简单说一下 拯救即将崩溃代码–iOS的消息转发

动态绑定引发

因为OC是一个动态运行时语言,其中之一的特性就是动态绑定。

关于动态绑定,苹果官网的给的解释为:(determining the method to invoke at runtime)。

简单点来说就是:程序直到执行时才能确定实际要调用的方法。

这样就会造成一个问题,我可以向一个实例发送一个消息,让它执行一个不属于自己的方法。这个时候就会出现unrecognized selector sent to instance

如果发生这种情况,那么我们就可以应用消息转发来解决这个问题。把这个不属于自己的方法变成属于自己的方法,或者找一个有这个方法的实例来执行这个方法。

拯救即将崩溃代码

第一步 方法解析处理阶段 | 动态方法决议

该方法内可为当前类动态添加方法。

将sel的方法实现指向一个已存在的方法


/**
 方法解析处理阶段 | 动态方法决议
 该方法内可为当前类动态添加方法。
 将sel的方法实现指向一个已存在的方法
 */
+ (BOOL)resolveInstanceMethod:(SEL)sel
{
    NSString *selectorString = NSStringFromSelector(sel);
    printf("%s %s\n", __func__, selectorString.UTF8String);
    
    // 根据 sel 得到 class 的实例方法
    Method method = class_getInstanceMethod([self class], @selector(dynamic_method));
    // 根据 sel 得到 class 的函数指针
    IMP method_imp = class_getMethodImplementation([self class], @selector(dynamic_method));
    // 给找不到实现的sel添加实现
    BOOL ret = class_addMethod([self class], sel, method_imp, method_getTypeEncoding(method));
    printf("%s\n", ret?"交换添加成功":"交换添加失败");
    // 返回结果不影响流程
    return YES;
}

第二步 快速转发阶段 | 快速查找

上一步未解决问题时触发。

返回一个能响应aSelector的实例,即将aSelector转发给另外的类。

/**
 快速转发阶段 | 快速查找
 上一步未解决问题时触发。
 返回一个能响应aSelector的实例,即将aSelector转发给另外的类。
 */
- (id)forwardingTargetForSelector:(SEL)aSelector
{
    NSString *selectorString = NSStringFromSelector(aSelector);
    printf("%s %s\n", __func__, selectorString.UTF8String);
    
    if ([selectorString isEqualToString:@"no_imp_method"]) {
        // 返回一个实现了aSelector函数的实例
        // 如果该实例没有实现aSelector,则进入下一步methodSignatureForSelector
        printf("%s 转发消息至BackUpClass\n",__func__);
        return  [[BackUpClass alloc] init];
    }
    // 返回self或者nil,则说明没有可以响应的目标,则进入下一步methodSignatureForSelector。
    return nil;
}

第三步 常规转发阶段 | 慢速查找

获得一个方法签名。签名由一个能响应aSelector的实例生成。
有签名则进入消息转发的最后一步forwardInvocation。

/**
 常规转发阶段 | 慢速查找
 返回一个方法签名。签名由一个能响应aSelector的实例生成。
 
 */
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector
{
    NSString *selectorString = NSStringFromSelector(aSelector);
    printf("%s %s\n", __func__, selectorString.UTF8String);
    
    BackUpClass * backUp = [BackUpClass new];
    NSMethodSignature * sign = [backUp methodSignatureForSelector:aSelector];
    //有签名则进入消息转发的最后一步forwardInvocation
    return sign;
}

也可以什么都不处理,至此本次消息转发结束,也不会crash。

/**
 将sel转发给一个真正实现了sel的对象
 也可以什么都不处理,至此本次消息转发结束,也不会crash。
 */
- (void)forwardInvocation:(NSInvocation *)anInvocation
{
    printf("%s %s\n",__func__ , anInvocation.description.UTF8String);
    
    // 创建备用消息接收对象
    BackUpClass * backUp = [[BackUpClass alloc] init];
    printf("%s 转发消息至BackUpClass\n",__func__);
    [anInvocation invokeWithTarget:backUp];
}

写在后面

浅谈。可以交流。

上一篇:3、Mybatis源码剖析


下一篇:SpringBoot的Web功能