消息转发

如果一直到root class都没有定位到SEL的实现,那么转入消息转发过程。

消息转发_第1张图片

消息转发流程图

步骤一:动态解析

通过实现resolveInstanceMethod:/resolveClassMethod:方法,我们有机会为该未知消息(SEL)新增一个“处理方法”(IMP)。

这意味着在消息转发前,你有机会通过class_addMethod给类动态添加一些方法

实际上返回值YES/NO无关紧要,只要你在resovle过程中新增过方法,就会触发class_getMethodImplementation,其作用相当于重新启动一次消息发送过程。

步骤二:备用接收者

通过实现-forwardingTargetForSelector:方法将消息(SEL)直接转发给另一个对象(备用接收者),也就是在另一个对象(不能是nil或self)上重启消息发送过程。

步骤三:完整转发

通过实现-methodSignatureForSelector:提供方法签名(即参数和返回值的类型信息)

可通过调用其他类的+instanceMethodSignatureForSelector:方法或其他对象的-methodSignatureForSelector:方法提供

也可通过+signatureWithObjCTypes:自行生成

生成的签名将和原始消息一起打包到一个NSInvocation对象中。

通过操作NSInvocation对象的target、selector属性可以方便地转发,甚至转发给另一个对象的另一个需要不同参数的SEL也是可以的

通过-getArgument:atIndex:和-setArgument:atIndex:可以操作方法调用传入的参数

通过-getReturnValue:和-setReturnValue:可以直接操作方法invoke后的返回值。

实现-forwardInvocation:方法

通过调用-invoke方法重新启动一个消息发送过程。

不调用invoke,吞掉这个消息(不做任何处理)

3 转发的功能

转发和多继承

消息转发_第2张图片

转发模拟了继承,所以可以用来为Objc程序提供类似多继承的功能。转发和多继承的区别如下:

◈ 多继承是将许多功能combine到一个对象中;

◈ 转发则将功能分解到多个对象,并一种对消息发送者透明的方式将它们关联起来;

代理/替代对象

场景描述:当你有一个对象,这个对象的设置由于需要处理大量数据非常耗时,所以更倾向于懒加载——在真正需要或系统空闲的时候来进行加载,这时你需要一个占位对象来使得应用的其他部分正常工作,这个占位对象的工作如下:

◈ 获取关于待加载数据的描述信息

◈ 转发消息时检测对象是否创建并已加载完数据,据此决定创建对象、丢弃消息或转发消息。

转发和继承

以下方法只考虑类继承体系(不含转发链);如需要对象表现得和继承一样,重写它们并把转发算法包括进来:

◈ -respondsToSelector: & +instancesRespondToSelector:

◈ -isKindOfClass: & -isMemberOfClass:

◈ -conformsToProtocol:

总结:

消息机制:继承体系搜索 -> 消息转发 ( 动态解析-> 快速转发 -> 完整转发 )

转发和继承(-respondsToSelector:等)、多继承、代理

你可能感兴趣的:(消息转发)