美文网首页
iOS开发之Runtime学笔记

iOS开发之Runtime学笔记

作者: 东也_ | 来源:发表于2018-12-12 20:05 被阅读60次

    runtime的数据结构

    🔘objc_object == id 所有对象 是一个结构体 

    🔘objc_class == Class 继承自objc_object 也是一个结构体 也叫类对象

    🔘isa指针实际上是C++中的共用体在runtime中定义为isa_t   

         isa指针又分为: 指针型isa 和 非指针型isa 

        🔘指针型isa 里面存放的只有class的地址

        🔘非指针型isa 在64位系统架构中,一般class的地址只需要30-40位就可以存放地址值,剩余的部分可以存放其他东西以达到节省内存的作用。

        

    🔘cache_t 

        🔘用于快速查找方法执行函数 

        🔘是可增量扩展的哈希表结构 

        🔘是局部性原理的最佳应用(局部性原理是将那些调用方法频率高的方法放到缓存列表)

    🔘class_data_bits_t 存放类的 属性 方法列表 

        

    🔘method_t 结构体 {

        SEL - > 函数名称 ;

        const char* types -> 包含返回值和形参;

        IMP函数指针 ->函数体 ;

        }

    实例对象、类对象和元类对象

        实例对象是objc_object, 里面就一个isa(objc_class)

        类对象中存储实例方法列表等信息  

        元类对象中存放类方法列表等信息

    实例对象的isa都指针指向类对象 类对象的superclass指向父类 根类对象指向nil

    类对象的isa指向元类对象 元类对象的superclass指向父类 根元类对象的superclass是指向根类对象

    元类对象的isa指针都指向它的根元类对象,而根元类对象的isa指针就指向它自己 

    实例对象类对象元类对象结构图

        调用实例对象方法的消息传递过程:

           根据当前实例对象的superclass指针找到类对象;在类对象中的方法列表中找到该方法实现;如果没有,就到类对象的superclass寻找直到根类对象;如果到根类对象还没有找到相应的方法实现就会产生崩溃。

        调用实例对象的类方法的消息传递过程:

            根据当前实例对象的isa找到类对象;然后通过类对象的isa指针找到元类对象;如果该元类对象没有该类方法实现,就会到元类对象的superclass中查找直到根元类对象;如果元类对象还没有的话,就会到根类对象中去查找同名实例方法,因为根元类对象的superclass是指向根类对象的;最后根类对象中也没有该方法,就凉凉了。

    消息传递的过程:

    objc_msgSend(id,SEL);类对象中查找

    objc_msgSendSuper(id,SEL);跨越当前实例对象的类对象,从类对象的superclass开始查找

    1、查看缓存是否存在方法实现(哈希查找) 

    2、没有去父类查找(对已排序好的方法列表进行二分查找,未排序的使用遍历查找) 

    3、 逐级查找直到nil(在每个类中都会经历前两步) 

    4、如果根类都没有就会进入消息转发流程。

    消息传递流程

    方法缓存查找的步骤:

           在cache_t中采用哈希算法进行查找,找到对应的bucket_t中的IMP函数体;

    消息转发的流程:

    消息转发流程图

        在消息传递过程中,未找到相应方法实现就进入到消息转发流程:

        1、是否处理;调用该类的类方法resolveInstanceMethod:(如果调用的是类的类方法就调用类的resolveClassMethod:);

        2、如果处理 结束转发流程;如果不处理,调用forwardingTargetForSelector返回转发的目标对象,也就是让另外一个对象去处理,然后结束转发流程。

        3、如果返回nil,就进入下一个流程,调用methodSignatureForSelector:返回方法签名对象NSMethodSignature,在调用forwardInvocation:执行;

        4、如果方法签名返回nil或者forwardInvocation处理不成功,就会产生崩溃,也就是常见的错误‘unrecognized selector’;

    动态方法解析

    @dynamic定义属性是 在动态运行时为这个属性添加set 和get方法。

    相关文章

      网友评论

          本文标题:iOS开发之Runtime学笔记

          本文链接:https://www.haomeiwen.com/subject/hdavxqtx.html