消息转发

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

消息转发流程图

步骤一:动态解析

通过实现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 转发的功能

转发和多继承

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

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

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

代理/替代对象

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

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

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

转发和继承

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

◈ -respondsToSelector: & +instancesRespondToSelector:

◈ -isKindOfClass: & -isMemberOfClass:

◈ -conformsToProtocol:

总结:

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

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

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

推荐阅读更多精彩内容

  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,840评论 0 9
  • 对象在收到无法解读的消息会发生什么? 若想令类理解某条消息,必须有对应的实现方法.但是,在编译期间动态像对象添加方...
    有一种再见叫青春阅读 598评论 0 2
  • 最近开始学习Runtime相关的内容.之前的话知道OC是一门动态的语言(动态类型,动态绑定,动态加载),却不知道到...
    kikido阅读 1,850评论 0 1
  • 消息转发三部曲: 接上面消息发送,如果当前类和父类中都没有找到实现,那么就会开始尝试动态方法解析。 动态方法解析 ...
    s_在路上阅读 2,203评论 2 14
  • 书上说当你处在低谷时不妨多出来走走看看,于是我便像被放风一般从自我的世界里走了出来,一切都是美好的,一切都是...
    红豆面包淡咖啡阅读 137评论 0 0