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];
}

写在后面

浅谈。可以交流。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 消息机制(Messaging) 不知大家有没有想过:我们在程序中调用的方法,是怎么执行的,又是怎么通过一个方法名字...
    百客阅读 1,373评论 0 8
  • 转载自:OC学习Runtime之消息传递,消息转发机制 参考文献:Effective Objective-C No...
    路漫漫其修远兮Wzt阅读 281评论 0 1
  • 一、简介: 消息转发是OC底层一种功能强大的实现,为OC方法的调用增加更多的表现力和容错能力。什么是消息转发?简单...
    司空123阅读 1,268评论 0 2
  • 消息转发(Message Forwarding)是在查找 IMP 失败后执行一系列转发流程的慢速通道,如果不作转发...
    二猪哥阅读 3,935评论 0 7
  • 我是黑夜里大雨纷飞的人啊 1 “又到一年六月,有人笑有人哭,有人欢乐有人忧愁,有人惊喜有人失落,有的觉得收获满满有...
    陌忘宇阅读 8,615评论 28 53