对象的基础数据结构

在WWDC20中,苹果官方介绍了对类的修改,其中出现了一个新的结构 class_rw_ext_t

首先,我们先来了解下,clean memory 与 dirty memory的区别:

clean memory : 加载后,不会被修改。在内存紧张时,可以从内存中移除,需要时再次从文件中加载。 (如class_ro_t)

dirty memory : 加载后,会被修改,一直存在于内存中,所以dirty memory更加”宝贵“

在旧版本的runtime中,类一旦被加载到内存中,就会初始化一个rw结构,并将所有的ro中的methods,protocols,properties复制到rw中,rw属于dirty memory。但大约90%的类并不需要对rw中的methods等进行修改,所以这部分”dirty memory“无疑是浪费的。所以苹果对class_rw_t进行进一步拆分出class_rw_ext_t用来存储这部分可能被修改的methods等,而 class_rw_ext_t只有在需要的时候,才被创建。而不被创建的class_rw_ext_t则是优化后,相对于旧版runtime可以节省下的内存空间。(相当于懒加载)

旧runtime
WeChat6e1e88b9b27b55efe3ad585156fb2483.png

首先从objc_class这么一个结构体(数据结构)开始,objc_class继承于objc_object。
objc_object当中有一个成员变量叫isa_t,那么这个isa_t指针就指向一个objc_class类型的类对象(或者说元类对象)。
objc_class主要包含三个成员变量,superClass(指向当前类的父类)、cache_t(用来提供消息传递过程中的缓存方法查找)、class_data_bits_t(类的一些基本信息:类所定义/通过分类所添加 的成员变量,属性,方法列表都在这个数据结构中)
superClass:实际是class类型的,它指向objc_class类型的这样一个指针
cache_t:实际上是装满了bucket_t数据结构的Hash表
class_data_bits_t:实际上是对class_rw_t的数据结构的封装 class_rw_t中包含了(class_ro_t 类相关的只读信息、protocols 类分类中的协议、properties 类分类中的属性、methods 类分类中的方法)
class_ro_t包含了name类名、methodList类的方法列表 --method_t、ivars声明的类的成员变量、类的属性、类的协议。

新版runtime

class_rw_t 新增了 class_rw_ext_t


WeChat21e051f62979fe9a2e3188eda5179a08.png

    const method_array_t methods() const {
        auto v = get_ro_or_rwe();
        if (v.is()) {
            return v.get()->methods;
        } else {
            return method_array_t{v.get()->baseMethods()};
        }
    }

旧版本runtim直接访问rw的methods属性,查找方法。在类加载到内存时,直接copy了ro的方法列表,所以无论方法列表是否被改变,这块内存都会开辟。

新版取方法、协议、属性是通过 method 方法获取的 如果class_rw_ext没创建 则从取class_ro_t

总结:新版的rumtime利用了懒加载的机制,在类的methods,properties等需要修改时,才初始化class_rw_ext_t这块“dirty+memory”存储这些列表,这样就减少了在旧版rumtime中90%的类在rw中直接复制ro中数据浪费的内存。

何时加载的?
1.装载分类时
2.runtime动态添加方法时
3.runtime添加property时
4.runtime添加protocol时

你可能感兴趣的:(对象的基础数据结构)