关于runtime的消息机制,你看我就够了!

31 篇文章 0 订阅

在上一个博客我们介绍了,消息机制的第一个阶段 超级详细的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个阶段都是一样的,我们就答出我们理解的三个阶段即可!

接下来博客我会介绍runtime的其他知识点,来继续学习runtime.

如果觉得我写得对您有所帮助,请关注我,我会持续更新😄

  • 5
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值