在上一个博客我们介绍了,消息机制的第一个阶段 超级详细的Runtime的消息机制的消息发送阶段,现在我们继续来探究消息机制的的第二三阶段.
消息机制发送阶段的回顾
之前的一幅图标记的很清楚,这里我就用文字叙述一下:
1.首先会判断传入的receiver是不是为空,为空就退出,如果不为空继续执行
2.再从receiverClass的cache中查找方法,如果找到方法就调用结束,找不到方法就会继续从receiverClass的class_rw_t中的方法列表中查找方法,如果找到就调用结束退出,并把方法缓存到receiverClass的cache中;如果找不到就继续就会继续在superClass的cache中查找,找到以后就会调用方法,并将方法缓存到receiverClass的cache中,找不到就会继续调用superClass的class_rw_t的方法列表中去找,同样的找到直接调用,并把方法缓存到receiverClass的cache中,如果找不到,就会继续找父类...直到最后还是还是找不到,就会继续执行第二阶段;
消息机制的第二阶段:动态方法解析
先看源码(源码的查找,上个博客已经说得很清楚了,直接接着上个博客说)
继续查找resolveMethod_locked的源码实现,如下图
图5好,我们现在就用代码验证一下是不是对象方法就会调用resolveInstanceMethod方法(类方法同理)
请看下面的代码
我们直接看实现
这里就是成功的把test的方法通过runtime动态方法解析到other上吗,至于类方法就会调用+(BOOL)resolveClassMethod:(SEL)sel一样的道理,大家可以尝试一下,现在我们就能总结了
从上面的源码分析,动态解析阶段只会once一次,上面写了也确实只会执行一次,当动态方法解析结束以后第二阶段才算完成
消息机制的第三阶段:消息转发
我们从上面的图5我们知道,第二阶段执行完成以后还会在回掉之前的方法,就会进入其中一个消息转发,我们看一下源码如下图
接下来我们看看forward_imp的定义
接下来我们看看_objc_msgForward_impcache的定义,接着你会发现又找到汇编去了
记得找arm64的哟,然后接着找__objc_forward_handler在当前文件,发现找不到,就全局搜索(注意一个是_一个是__,这个前面已经解释了).
这里只能看到打印,看不出其他的操作,其实到这里说明消息转发的具体代码是闭源的了,我们找不到实现,其实也能在网上找一些伪代码看看是怎么实现的,不过没关系,我们接下来直接说结论
好了接下来我继续创建一个类,叫做GDCat,GDCat里面也定义了一个test方法,里面写了实现,我们请看下面的代码:
当第一阶段和第二阶段都没有成功的时候,这是就会进入消息转发,如上图,消息转发是不是很神奇,它可以转给另一个类调用test方法,如上图,这里我们要知道一个知识点,就是return [[GDCat alloc]init]; 这个就相当于 objc_msgsend([[GDCat alloc]init],@selector(test)),是不是就是这个道理,
如果-(id)forwardingTargetForSelector:(SEL)aSelector你没有返回对应的,这时候我们还有最后一次补救措施,就是还会在执行,请看下面的代码
对了如果[NSMethodSignature signatureWithObjCTypes:"v@:"];不熟悉
可以这么写: return [[[GDCat alloc]init] methodSignatureForSelector:aSelector];效果一样的,大家可以试试
这里有一个小插曲哈,请看下面的代码,如果签名返回的是空
这样返回报错是肯定的,只是之前的版本报错是:doesNotRecognizeSelector:会报这个错误,现在版本不报这个错误了,因为有些面试题会问,runtime的消息机制,什么时候会报doesNotRecognizeSelector这个错误,这个就是注意一下就行了,我们继续
接下来我们总结一下第三阶段:
注意:
细心的可能发现,上图中我并没有写+还是-方法,因为这里还有个注意点,就是对于消息机制,网上很多版本说得是,消息机制对于类方法是没有第三阶段的,这个是错误的,因为你直接写可能没有提示,所以很多人认为是没有第三阶段,其实是有的,请看下面的代码
再看下另一个:
所以可以看出,类方法也有第三阶段的消息机制的;这里我们就可以在forwardInvocation里面想做什么事就做什么事,比如你想改变方法,想改变返回值等等一系列的操作.这里我就不写了,有兴趣的可以一起讨论😄
runtime面试题
1.说一下OC的消息机制
OC中的方法调用其实都是转成了objc_msgSend函数的调用,给receiver(方法调用者)发送了一条消息(selector方法名字)
objc_msgSend底层有三大阶段....这个就不说了,上面叙述的非常清楚
2.说一下消息转发机制
就是我们上面说的第三个阶段,写得也是非常仔细,
等等就我们上面说的三大阶段,会有很多问题,等等
总结:
消息机制的三大阶段就是上面这么多,有些把这些分成2个阶段或者4个阶段都是一样的,我们就答出我们理解的三个阶段即可!
网友评论