WWDC20中objc增加的结构class_rw_ext_t

前言

        在WWDC20中,苹果官方介绍了对类的修改,其中出现了一个新的结构 class_rw_ext_t,那么为什么会出现这个东东呢?在看这篇文章时,希望你是阅读过runtime源码的。

Why?

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

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

        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可以节省下的内存空间。

2020前 rw结构
2020后rw结构

How?

        以“查找方法”为例,对新旧版本runtime进行比较:

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

旧版runtime查找方法代码


旧版runtime class_rw_t结构

        新版runtime则是访问rw的methods()成员方法来获取类的方法列表,而且在methods方法中判断了,rw_ext_t是否已经初始化了,如果初始化,则读取class_rw_ext_t的methods,否则从ro的baseMethods()获取类的方法列表。看下面的代码:


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


补充:那class_rw_ext_t内存又是在何时开辟的呢?

        1.装载分类时

        2.runtime动态添加方法时

        3.runtime添加property时

        4.runtime添加protocol时

你可能感兴趣的:(WWDC20中objc增加的结构class_rw_ext_t)