runtime:消息传递过程解析

今天遇到了一个引人第三方崩溃的问题,崩溃的信息是-[NSURLResponse statusCode]: unrecognized selector sent to instance 0x15785f230',趁机学习一下runtime的消息传递机制我把每次防止崩溃的时刻成为转机~,宗旨就是根据SEL找到IMP(让方法找到实现)。

根据obj_msgSend方法的调用顺序,做了如下的尝试:
1.首先写了一个NSURLResponse的category,因为NSURLResponse是个系统的类,无法在.m文件修改内容:

在runtime时,没有找到SEL的IML时会执行resolveInstanceMethod。利用class_addMethod添加函数

在这次转机中,还是很好理解的,特别需要注意的是resolveInstanceMethod中包含的IMP,实现方法是个C语言函数,该函数中必须包含有id和SEL两个参数。

2.如果resolveInstanceMethod返回的是NO,便会拐个弯继续根据SEL去找IMP。根据函数的名字forwardingTarget。。就是改变target。。改变目标。。

forwardingTargetForSelector返回的是一个其他对象。

在这次转机中,看来这个类已经对自己失望了,所以要去寻找别的类对象是否有对应的函数实现。

2'' 在下面的例子中,我尝试在forwardingTargetForSelector中返回是对象本身,在对象内部写了个函数,就好像对象自己回心转意了一下~~,还是可以执行的。我就暂且把它称为回旋转机吧。这样做其实是不和常规的,完全可以在第一次转机的时候。这里只是做个测试~

forwardingTargetForSelector返回的依然是其自身

3.methodSignatureForSelector应该是让对象最桑心的选择了,连其他对象都没有想要的函数,只能把函数的签名扔到茫茫人海中了。。就是将函数名称包装成签名NSMethodSignature。方法签名仅包含了参数类型和返回值。所以人家还需要调用forwardInvocation的调用才行。

methodSignatureForSeletor和forwardInvocation配合使用

在这次转机中,forwardInvocation函数内部,可以有很丰富的封装。可以拿到函数名,获取参数,设置返回值等。


另一方对象的实现

以上所写的内容,是为了屏蔽掉崩溃的问题。那么在平时的使用中,如果是想动态实现函数,还需要重载respondsToSelector中,将想要动态实现的函数返回为yes。不然调用respondeToSelector时,会返回NO。

objc_msgSend的调用过程如下:

0.检测这个selector是不是要忽略的。()

1.首先检测消息的接受者是否为nil,如果是nil,就不再发送消息了,所以nil对象执行任何一个操作都不会崩溃。(插曲:曾经想要写一个字符串的扩展,如果对象是nil,就输出孔字符串@“”。但是怎么也不生效。最后才恍然大悟,对象是nil,连消息都接收不到,怎么可能还变成@“”。现在想想也是蛮有趣的。)
///////////////////////////////////////////////////////////////////////////////////
如果是实例方法:
2.检查class的缓存中是否有这个方法的实现。如果有,就调用。(struct objc_cache *cache)指向最近使用的方法的指针,提高效率
3.若找不到,就查找方法列表。(struct objc_method_list **methodLists)方法地址列表,如CLS_CLASS (0x1L),则存储实例方法,如CLS_META (0x2L),则存储类方法;
4.查看父类中的methodLists是否有该方法的实现,如果没有就再查找父类的父类中是否有该方法。知道NSObject的根类为止。
///////////////////////////////////////////////////////////////////////////////////
如果是类方法:
2.会通过自己的isa找到自身类的metaClass(metaClass是用来存放类本身的,类方法就存储在metaclass中的methodList中),在methodList中查找
3.若找不到,就去查找metaClass的superClass,查找他的superClass(也是metaClass)的methodList中是否有对应函数。
////////////////////////////////////////////////////////////////////////////////////

5.调用resolveInstanceMethod:(或者resolveClassMethod:)如果有,就返回yes。并调用
6.调用forwardingTargetForSelector函数,如果返回的对象中包含该方法,调用
7.调用methodSignatureForSelector函数,如果返回了一个函数的签名,再调用forwardInvocation。根据里面的函数,调用即可。
8.调用doesNotRecognizeSelector。抛出异常

注意metaclass的isa直接就是metaclass的rootclass了。



消息传递机制

参考文章:

http://www.cnblogs.com/biosli/p/NSObject_inherit_2.html

http://my.oschina.net/u/566401/blog/182520

http://www.henishuo.com/runtime-message-forwarding/

http://blog.sunnyxx.com/2014/11/06/runtime-nuts/

最后,推荐一个我自己的产品,找到我啦,可查看轨迹和定位,欢迎关注我的微信公众号,时刻关注找到我啦的更新


推荐阅读更多精彩内容

  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,226评论 0 9
  • 前言 runtime其实在我们日常开发过程中很少使用到,尤其是像我现在比较初级的程序猿就更用不到了。但是去面试很多...
    WolfTin阅读 375评论 0 2
  • 这篇文章完全是基于南峰子老师博客的转载 这篇文章完全是基于南峰子老师博客的转载 这篇文章完全是基于南峰子老师博客的...
    西木阅读 29,526评论 33 467
  • 一、Runtime简介 Runtime简称运行时。OC就是运行时机制,也就是在运行时候的一些机制,其中最主要的是消...
    林安530阅读 679评论 0 2
  • 运行时是iOS中一个很重要的概念,iOS运行过程中都会被转化为runtime的C代码执行。例如[target do...
    萝卜酱紫阅读 162评论 0 3
  • 关于亲子阅读的方式 前段时间跟一个朋友聊天,她是一个男孩儿的妈妈,因为老公平时里工作比较忙,她总是纠结于怎么开展亲...
    taku麻麻阅读 62评论 0 0
  • 第一章 《缪境》(4) “在无法得知事实真相的情况下,我不排除任何可能性。”我说。 “你现在想问题的方式变得和修泽...
    阿尔法零阅读 133评论 0 6
  • 霞是故乡美 她映红了垂暮的天空 霞是故乡美 她藏进了踱步的云中 霞是故乡美 她装扮了穹起的峦嵘 霞是故乡美 她变作...
    隔壁Uncle王阅读 110评论 1 1
  • 熟悉的街景 寻不到的身影在远方前行 挥手作别的场景 坦言于酒后的真情 转身将泪水放行 梦中响起的歌声 谁将我唤醒:...
    雲墨寒阅读 102评论 0 2
  • 秒针开始滴滴答答的逆时针行走 昼开始退回成夜,再退回成昼 夕阳从西方升起,落向东方 落叶开始变绿,从地面飘回枝头 ...
    春秋夏冬阅读 156评论 0 0