iOS之消息转发机制

        在编译期,向对象或者类对象发送了其无法解读的消息并不会报错,因为在运行期可以继续向类和元类中添加方法,所以编译器在编译期还无法确定类中到底会不会有某个方法的实现。当对象接收到无法解读的消息后,就会启动“消息转发”机制,我们可以在消息转发过程中告诉对象应该如处理未知消息。下面让我们来先学习下消息转发流程。

        消息转发分为两个阶段。第一阶段叫做“动态方法解析”,或叫“动态方法决议”。第二阶段涉及到“完整的消息转发机制”,或者叫“完整的消息转发原理”。

动态方法解析

        动态方法解析的意思就是征询消息接受者所属的类,看其是否能动态添加方法,以处理当前“这个未知的选择子“。对象在收到无法处理的消息时,会调用下面的方法:

//未实现实例方法调用

+ (BOOL)resolveInstanceMethod:(SEL)selector

//未实现类方法调用

+ (BOOL)resolveClassMethod:(SEL)selector

        如果在运行期系统已经执行完了动态方法解析,那么消息接受者自己就无法再以动态新增方法的形式来响应包含该未知选择子的消息了,此时就进入了第二阶段——完整的消息转发。运行期系统会请求消息接受者以其他手段来处理与消息相关的方法调用。

完整的消息转发机制

        完整的消息转发又分为两个阶段,第一阶段称为备援接受者,第二阶段才是启动完整的消息转发机制。

备援接受者

        当前接受者如果不能处理这条消息,运行期系统会请求当前接受者让其他接受者处理这条消息,与之对应的方法是:

- (id)forwardingTargetForSelector:(SEL)selector

        在这个方法里,方法参数代表未知的选择子(方法),返回值为一个能够处理该消息的对象(备援接受者),若当前接受者能找到对象,就直接返回,这个未知的选择子将会交由该对象处理。如果找不到对象,就返回nil,此时就会启用”完整的消息转发机制“。

完整的消息转发机制

        如果转发算法已经到了这一步的话,那么唯一能做的就是启用完整的消息转发机制了。完整的消息转发机制是这样的:首先创建NSInvocation对象,把尚未处理的那条消息有关的全部细节封装于这个NSInvocation对象中。此对象中包含选择子、目标及参数。在触发NSInvocation对象时,”消息派发系统“将亲自触发,把消息派发给目标对象。

- (void)forwardInvocation:(NSInvocation *)invocation

        对于类似于unrecognized selector send to instance xxx这样的错误,你可能觉得很熟悉。这种错误通常是因为调用了某个对象或者某个类里不存在的方法,从而触发了消息转发机制,最终把这个未识别的消息发送给了NSObject的默认实现。

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

推荐阅读更多精彩内容

  • 一>形成由来:Object-C是在C的基础上进行编写的一门动态程序语言(也称超C)。底层全部是由C语言实现的。所谓...
    silencerZiBo阅读 4,798评论 0 2
  • 在编译期向某类发送了其无法理解的消息并不会报错,因为在运行期可以继续向类中添加方法,所以编译器在编译时还不确定类中...
    修炼一颗真心阅读 2,953评论 0 0
  • Objective-C是使用传递消息的机制来调用函数,这就会使用到动态绑定的机制在运行期来决定到底调用哪个方法,甚...
    学习的途中阅读 3,392评论 0 1
  • 函数的调用方式 Objective-C是C语言的超集,C语言的函数调用方式是“静态绑定的”,也就是说在编译的时候就...
    littlewish阅读 5,646评论 3 8
  • 前言 之前对于iOS的消息转发机制只是有个大致的了解,最近刚好有空,总结下相关的知识点。 1、函数的调用方式 Ob...
    WQ_UESTC阅读 3,751评论 0 5