#Runtime

作者: Oh丶Really | 来源:发表于2020-07-23 01:07 被阅读0次

    包括以下内容

    • 数据结构
    • 类对象与元类对象
    • 消息传递
    • 方法缓存
    • 消息转发
    • Method-Swizzling
    • 动态添加方法
    • 动态方法解析

    1.数据结构

    • objc_objct
    • objc_class
    • isa指针
    • method_t

    objc_object:

    image

    objc_class:

    image

    isa指针:

    image
    • 对象的isa指针指向其类对象
    • 类对象的isa指针指向其元类对象

    cache_t:

    可以理解为一个数组,其元素是 bucket_t 结构体
    bucket_t 由 key 和 IMP 组成;key 是方法的名称,IMP是方法的实现

    • 用于快速查找方法执行函数
    • 是可增量扩展哈希表
    • 局部性原理的最佳应用

    class_data_bits_t:

    • class_data_bits_t主要是对class_rw_t的封装
    • class_rw_t代表了类相关的读写信息、对class_ro_t的封装
    • class_ro_t代表类相关的只读信息
    class_rw_t:
    image
    class_ro_t:
    image
    method_t:
    image
    const char * types -------Type Encodeing
    image

    数据结构 总结

    image

    2.对象、类对象、元类对象

    • 类对象--存储实例方法列表等信息
    • 元类对象--存储类方法列表等信息
    image

    类对象与元类对象都是 objc_class 结构,又 objc_class 继承自 objc_object 。所以,都有 isa 指针,实例对象的 isa 指针指向其类对象,类对象中存储实例方法类表等信息;类对象的 isa 指针指向其元类对象,从而可以访问类方法列表等信息;元类对象的 isa 指针都指向其根元类对象;根元类对象的 superclass 指向类对象-----(所以若一个元类方法中没找到某类方法,会查找其实例方法代替)。

    3.消息传递

    提问,打印的内容
    image
    分析
    image image

    那msgSend 和 msgSendSuper这两者有什么区别呢?

    消息传递机制
    image

    所以,本题打印的都为 phone

    1、缓存查找

    采用哈希查找,为了解决查找效率

    例:给定值是 SEL ,目标值是对应的 bucket_t 中的 IMP

    通过 key 和 mask 做位与操作,查找到对用 bucket_t 的索引位置

    2、当前类中的查找

    当前类中的方法一般有方法列表:

    • 对于已排序号的列表,采用二分查找算法查找对应的执行函数;
    • 对于没有排序的列表,采用一般遍历查找对用函数

    3、父类逐级查找

    image

    4.消息转发

    消息转发流程


    image

    5.Method-Swizzling

    image

    6.动态添加方法

    问题:你是否使用过 performSelector 方法?
    分析:可能考察运行时,可能是一个类在编译时没有这个方法,在运行时产生了这个方法。
    image

    7.动态方法解析

    问题:是否使用过 @dynamic
    分析:当属性为 @dynamic 时,相当于说这个属性的 set 和 get 方法是在运行时而不是在编译时声明好具体的实现
    • 动态运行时语言将函数决议推迟到运行时(当调用到相关方法添加具体的执行函数)
    • 编译时语言在编译期进行函数决议(在编译时期就决议好具体的函数实现,运行时无法修改)

    练习

    • [obj foo] 和 objc_msgSend() 函数之间有什么关系?
    • runtime 如何通过 Selector 找到对应的 IMP 地址?
    • 能否向编译后的类增加实例变量?

    解答

    • --- 前者在编译器处理好会转化成后者消息发送
    • --- 查找当前实例类对象的缓存,若无,查找当前类的方法列表,若无,逐级查找父类的方法列表
    • ----不行,为 class_ro_t 类型,为只读的,不能添加。但是可以向动态添加的类添加实例对象,只要在调用注册信息之前完成实例变量的添加即可实现。

    相关文章

      网友评论

          本文标题:#Runtime

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