该文是 objc_msgSend消息传递学习笔记 - 对象方法消息传递流程 的基础上继续探究源码,请先阅读上文。
消息转发机制(message forwarding)
Objective-C 在调用对象方法的时候,是通过消息传递机制来查询且执行方法。如果想令该类能够理解并执行方法,必须以程序代码实现出对应方法。但是,在编译期间向类发送了无法解读的消息并不会报错,因为在 runtime 时期可以继续向类添加方法,所以编译器在编译时还无法确认类中是否已经实现了消息方法。
当对象接受到无法解读的消息后,就会启动消息转发机制,并且我们可以由此过程告诉对象应该如何处理位置消息。
本文的研究目标:当 Class 对象的 .h
文件中声明了成员方法,但是没有对其进行实现,来跟踪一下 runtime 的消息转发过程。于是创造一下实验场景:
同上一篇文章一样,定义一个自定义 Class DGObject
,并且声明改 Class 中拥有方法 - (void)test_no_exist
,而在 .m
文件中不给予实现。在 main.m
入口中直接调用该类某实例的 - (void)test_no_exist
方法。
动态方法解析
依旧在 lookUpImpOrForward 方法中下断点,并单步调试,观察代码走向。由于方法在方法列表中无法找到,所以立即进入 method resolve 过程。
runtimeLock.unlockRead()
是释放读入锁操作,这里是指缓存读入,即缓存机制不工作从而不会有缓存结果。随后进入 _class_resolveMethod(cls, sel, inst)
方法。
此方法是动态方法解析的入口,会间接地发送 +resolveInstanceMethod
或 +resolveClassMethod
消息。通过对 isa 指向的判断,从而分辨出如果是对象方法,则进入 +resolveInstanceMethod
方法,如果是类方法,则进入 +resolveClassMethod
方法。
而上述代码中的 _class_resolveInstanceMethod
方法,我们从源码中看到是如此定义的:
通过 _class_resolveInstanceMethod
可以了解到,这只是通过 +resolveInstanceMethod
来查询是否开发者已经在运行时将其动态插入类中的实现函数。并且重新触发 objc_msgSend
方法。这里有一个 C 的语法值得我们去延伸学习一下,就是关于关键字 __typeof__
的。__typeof__(var)
是 GCC 对 C 的一个扩展保留字(官方文档),这里是用来描述一个指针的类型。
我们发现,最终都会返回到 objc_msgSend
中。反观一下上一篇文章写的 objc_msgSend
函数,是通过汇编语言实现的。在 Let’s build objc_msgsend 这篇资料中,记录了一个关于 objc_msgSend
的伪代码。
在缓存中无法直接击中 IMP 时,会调用 class_getMethodImplementation
方法。在 runtime 中,查看一下 class_getMethodImplementation
方法。
在上一篇文中,详细介绍过了 lookUpImpOrNil
函数成功搜索的流程。而本例中与前相反,我们我发现该函数返回了一个 _objc_msgForward
的 IMP。此时,我们击中的函数是 _objc_msgForward
这个 IMP ,于是消息转发机制进入了备援接收流程。
Forwarding 备援接收
_objc_msgForward
居然可以返回,说同 IMP 一样是一个指针。在 objc-msg-x86_64.s
中发现了其汇编实现。
发现在接收到 _objc_msgForward
指针后,会立即进入 __objc_forward_handler
方法。其源码在 objc-runtime.mm
中。
在 ObjC 2.0 以前,_objc_forward_handler
是 nil ,而在最新的 runtime 中,其实现由 objc_defaultForwardHandler
完成。其源码仅仅是在 log 中记录一些相关信息,这也是 handler 的主要功能。
而抛开 runtime ,看见了关键字 __attribute__((noreturn))
。这里简单介绍一下 GCC 中的又一扩展 __attribute__机制 。它用于与编译器直接交互,这是一个编译器指令(Compiler Directive),用来在函数或数据声明中设置属性,从而进一步进行优化(继续了解可以阅读 NShipster __attribute__)。而这里的 __attribute__((noreturn))
是告诉编译器此函数不会返回给调用者,以便编译器在优化时去掉不必要的函数返回代码。
Handler 的全部工作是记录日志、触发 crash 机制。如果开发者想实现消息转发,则需要重写 _objc_forward_handler
中的实现。这时引入 objc_setForwardHandler
方法:
这是一个十分简单的动态绑定过程,让方法指针指向传入参数指针得以实现。
Core Foundation 衔接
引入 objc_setForwardHandler
方法后,会有一个疑问:如何调用它?先来看一段异常信息:
这个日志场景都接触过。从调用栈上,发现了最终是通过 Core Foundation 抛出异常。在 Core Foundation 的 CFRuntime.c 无法找到 objc_setForwardHandler
方法的调用入口。综合参看 Objective-C 消息发送与转发机制原理 和 Hmmm, What’s that Selector? 两篇文章,我们发现了在 CFRuntime.c 的 __CFInitialize()
方法中,实际上是调用了 objc_setForwardHandler
,这段代码被苹果公司隐藏。
在上述调用栈中,发现了在 Core Foundation 中会调用 ___forwarding___
。根据资料也可以了解到,在 objc_setForwardHandler
时会传入 __CF_forwarding_prep_0
和 ___forwarding_prep_1___
两个参数,而这两个指针都会调用 ____forwarding___
。这个函数中,也交代了消息转发的逻辑。在 Hmmm, What’s that Selector? 文章中,复原了 ____forwarding___
的实现。
Message-Dispatch System 消息派发系统
在大概了解过 Message-Dispatch System 的源码后,来简单的说明一下。由于在前两步中,我们无法找到那条消息的实现。则创建一个 NSInvocation 对象,并将消息全部属性记录下来。 NSInvocation 对象包括了选择子、target 以及其他参数。
随后,调用 forwardInvocation:(NSInvocation *)invocation
方法,其中的实现仅仅是改变了 target 指向,使消息保证能够调用。倘若发现本类无法处理,则继续想父类进行查找。直至 NSObject ,如果找到根类仍旧无法找到,则会调用 doesNotRecognizeSelector:
,以抛出异常。此异常表明选择子最终未能得到处理。
而对于 doesNotRecognizeSelector:
内部是如何实现,如何捕获异常。或者说 override 改方法后做自定义处理,等笔者实践后继续记录学习笔记。
对于消息转发的总结梳理
在 Core Foundation 的消息派发流程中,由于源码被隐藏,所以笔者无法亲自测试代码。倘若以后学习了逆向,可以再去探讨一下这里面发生的过程。
对于这篇文章记录的消息转发流程,大致如下图所示:
以上是对于 objc_msgSend 消息转发的源码学习笔记,请多指正。
参考资料
Let’s Build objc_msgSend
Hmmm, What’s that Selector?
NShipster __attribute__
Objective-C 消息发送与转发机制原理
若想查看更多的iOS Source Probe文章,收录在这个Github仓库中。