Runtime初涉之消息转发

作者: Mars飘殇 | 来源:发表于2015-09-23 21:43 被阅读796次

    与上次发文间隔也有很长一段时间了,由于各种原因,没有及时拟写文章(好吧,我又矫情了),话不多说,咱们进入正题。

    上一篇文章咱们提到了Runtime的消息传递机制,主要围绕三个C语言API来展开进行的。这篇文章我将从另外三个方法来描述Runtime中另一个特性:消息转发机制。

    一、消息转发机制

    当向某个对象发送一条消息时,若该对象的方法列表以及它相应继承链上的方法列表都无法找到以该消息选择子作为key的方法实现时,则会触发消息转发机制。

    1、动态方法解析

    + (BOOL)resolveInstanceMethod:(SEL)sel;

    首先,当接受到未能识别的选择子时,运行时系统会调用该函数用以给对象一次机会来添加相应的方法实现,如果用户在该函数中动态添加了相应方法的实现,则跳转到方法的实现部分,并将该实现存入缓存中,以供下次调用。

    2、备援接收者

    - (id)forwardingTargetForSelector:(SEL)aSelector;

    如果运行时在消息转发的第一步中未找到所调用方法的实现,那么当前接收者还有第二次机会进行未知选择子的处理。这时运行期系统会调用上述方法,并将未知选择子作为参数传入,该方法可以返回一个能处理该选择子的对象,运行时系统会根据返回的对象进行查找,若找到则跳转到相应方法的实现,则消息转发结束。

    3、完整的消息转发

    - (void)forwardInvocation:(NSInvocation *)anInvocation;

    当运行时系统检测到第二步中用户未返回能处理相应选择子的对象时,那么来到这一步就要启动完整的消息转发机制了。该方法可以改变消息调用目标,运行时系统根据所改变的调用目标,向调用目标方法列表中查询对应方法的实现并实现跳转,这种方式和第二步的操作非常相似。当然你也可以修改方法的选择子,亦或者向所调用方法中追加一个参数等来跳转到相关方法的实现。

    最后,如果消息转发的第三步还未能处理该未知选择子的话,那么最终会调用NSObject类的如下方法用以异常的抛出,表明该选择子最终未能处理。

    - (void)doesNotRecognizeSelector:(SEL)aSelector;

    下面附上完整的消息转发流程图:

    消息转发


    二、消息转发验证

    好了,看了那么多的理论知识,相比大家也已经累了,那我们用一个实例来具体说明Runtime的消息转发机制吧。我是传送门~~~

    首先新建一个工程,并在工程中添加Cat、Dog and Chicken三个类,并在每个类的.h文件中声明jump方法,在Cat.m文件中声明消息转发的第一步方法:resolveInstanceMethod: ,在该方法中动态添加jump方法的实现。

    消息转发第一步

    注:在第一步中动态添加方法实现用到了Runtime中的class_addMethod方法,该方法用以向该类的实例对象中添加相应的方法实现。

    然后在main.m文件中调用Cat实例的jump方法,就会看到在控制台打印出如下结果:

    消息转发第一步打印结果

    然后在Dog.m文件中验证消息转发第二步过程,为了让运行时系统能够运行到forwardingTargetForSelector:方法,我们先在resolveInstanceMethod:中返回NO,代码如下:

    消息转发第二步

    然后按照之前的样子,在main.m文件中让Dog也jump起来,运行之后打印结果如下:

    消息转发第二步打印结果

    最后我们来验证消息转发第三步骤的过程。

    在最后的Chicken.m文件中我们让前两步的方法分别返回NO和nil值,用以快速触发消息转发机制中的完整消息转发机制。在验证这一步中我们注意到,在调用forwardInvocation:方法之前我们需要实现methodSignatureForSelector:方法,并将相应选择子的描述返回。

    消息转发第三步

    这里我用了改变调用目标这种方式进行消息转发机制,至于改换选择子,读者可以自行尝试运行哈,具体我已在项目代码中写明,最终调用Chicken实例的jump方法,其打印结果如下:

    消息转发第三步打印结果

    明明分别调用了三个动物的jump,最后在控制台只看到了Cat一直在jump。。。

    如果有觉得上述我讲的不对的地方欢迎指出,大家多多交流沟通。

    相关文章

      网友评论

      • f946b0007750:实际开发中的应用场景是什么样呐,给描述下白。
        Mars飘殇:@静眠的地瓜 这个是帮助你学iOS开发理解用的,通过这个,你就能知道iOS中,消息在运行时的传递过程是怎样的,在实际开发中确实没怎么会用到Runtime的这一机制。
      • 0211359d9a52:写的不错哈 :smiley:
        Mars飘殇:@EEEEEE 过奖了,互相学习:blush:
      • Cedric_Jc:大神接着更新阿。更一本RunTime...
        Mars飘殇:@Cedric_Jc 大神不敢当,但会继续更新 :blush:
      • 96dfced25580:mars请问能解释一下文中的“v@:”,作为参数传入具体意思是什么吗?
        96dfced25580:嗯嗯,完全明白了,就是对应的参数类型有对应的字符,这些字符有详细的对照表么?想深入看看,anyway,感谢大神。
        Mars飘殇:@仲系xbb 该字符串是用于描述函数返回值类型和参数类型的字符串,其中v表示jump函数返回值是void类型(如果返回值为int类型,则为i),第二个字符@表示jump方法的第一个参数类型id,因为该参数是对象类型,所以用@,第三个字符:表示jump方法中的第二个参数类型SEL,若jump方法后续还有其他参数,则要在该字符串后面再添加用于描述相应参数类型的字符,希望我的回答能帮助你理解哈~。

      本文标题:Runtime初涉之消息转发

      本文链接:https://www.haomeiwen.com/subject/ygpmqttx.html