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的默认实现。

你可能感兴趣的:(iOS之消息转发机制)