2020 WWDC 大会 iOS 变化记录(一)—— 数据结构的变化

一、前言

去年 2020 年的 WWDC 大会时,因为我英语也不太好,我就边看录播边用谷歌翻译着,记录了一下这次会议的一些跟我们开发者有关的变化点。一直没有整理发出来,这次想着把这个记录发一下,不然 2021WWDC 大会都要开始了。
变化点主要有三点,这篇主要讲一下第一点,“数据结构的变化”(Class data structures changes)

二、数据结构的变化

1、"clean memory" 和 "dirty memory"

先理解两个名词,"clean memory""dirty memory",字面意思就是 “干净内存” 和 “脏内存”。

  • clean memory 是指加载后不会发生改变的内存。
    class_ro_t 就属于clean memory, 因为它是只读的(ro 代表 readonly
  • dirty memory 是指在进程运行时会发生更改的内存,class_rw_tdirty memory(rw 代表 read write)

objc4750 的源码中我们可以看到 class_rw_t 的结构,如下图:

class_ro_t 的结构,如下图:

类结构一经使用就会变成 dirty memory,因为运行时会向它写入新的数据,例如创建一个新的方法缓存并从类中指向它。
dirty memoryclean memory 要昂贵得多,只要进程在运行,它就必须一直存在。另一方面 clean memory 可以进行移除,从而节省更多的内存空间。因为如果你需要 clean memory,系统可以从磁盘中重新加载。macOS 可以选择换出 dirty memory,但因为 iOS 不使用 swap,所以 dirty memoryiOS 中代很大。

dirty memory 是这个类数据被分成两部分的原因,可以保持清洁的数据越多越好,通过分离出那些永远不会更改的数据,可以把大部分的类数据存储为 clean memory。虽然这些数据足以让我们开始,但运行时需要追踪每个类的更多信息,所以当一个类首次被使用,运行时会为它分配额外的存储容量,这个运行时分配的存储容量是 class_rw_t,用于读取-编写数据。在这个数据结构中,我们存储了只有在运行时才会生成的新信息。

例如,所有的类都会链接成一个树状结构,这是通过 First SubclassNext Sibling Class 指针实现的。这允许运行时遍历当前使用的所有类,这对于使方法缓存无效非常有用。
但为什么方法和属性也在只读数据中时,我们这里还要有方法和属性呢?

因为他们可以在运行时进行更改。当 category 被加载时,它可以向类中添加新的方法,而且程序员可以使用运行时 API 动态地添加它们。因为 class_ro_t 是只读的,所以我们需要在 class_rw_t 中追踪这些东西。
现在,结果是这样做会占用相当多的内存。在任何给定的设备中都有许多类在使用,我们再 iPhone 上的整个系统中测量了大约 30 兆字节这些 class_rw_t 结构,那么我们如何缩小这些结构呢?记住,我们在读取-编写部分需要这些东西,因为它们可以在运行时进行更改。但是通过检查实际设备上的使用情况,大约只有 10% 的类真正地更改了它们的方法。而且只有 Swift 类会使用这个 demangled name 字段,并且 Swift 类并不需要这一字段,除非有东西询问它们的 Object-C 名称时才需要。

2、变化 —— 拆出了 class_rw_ext_t

所以,我们可以拆掉那些平时不用的部分,这将 class_rw_t 的大小减少了一半。拆出来的部分叫做 “class_rw_ext_t”,如下图所示:

这个 “class_rw_ext_t” 我们可以在最新的源码 objc4-818.2 中看到:

struct class_rw_ext_t {
    DECLARE_AUTHED_PTR_TEMPLATE(class_ro_t)
    class_ro_t_authed_ptr<const class_ro_t> ro;
    method_array_t methods;
    property_array_t properties;
    protocol_array_t protocols;
    char *demangledName;
    uint32_t version;
};

对于那些确实需要额外信息的类,我们可以分配这些扩展记录中的一个,并把它滑到类中供其使用。

大约 90% 的类从来不需要这些扩展数据,这在系统范围内可节省大约14MB 的内存,这些内存现在可用于更有效的途径,比如存储你的 app 的数据。
因此,实际上你可以在你的 Mac 上看到这一变化带来的影响,这只需要在终端上运行一些简单的命令,现在让我们一起来看一下。
打开 MacBook 的终端,运行一个在任何 Mac上都可用的命令,叫做“heap”,它允许你检查正在运行的进程所使用的堆内存。我将在 Mac 中的 Mail app 上运行它,首先先打开 “邮件” 程序,

现在,如果我运行 heap Mail 命令,它会输出数千行,显示通过邮件进行的每个堆分配,所以我只要 grep 它为我们今天一直在讨论的 class_rw_t 类型,我还需要查询标头

heap Mail | egrep ‘class_rw|COUNT’

从返回的结果中,可以看到,我们在邮件 app 中使用了大概 9000 个这样的 class_rw_t 类型,但其中只有大约十分之一,900 多一点实际上需要使用这一扩展信息。所以,我们可以很容易地计算出通过这个改变所节省的内存,这是大小减半的类型。

所以,如果我们从这个数字 “152320” 中减去我们必须分配给扩展类型的内存量 ”21600”,我们可以看到我们节省了大约 1 兆字节数据的四分之一,这还只是对于邮件 app 而言。如果我们在系统范围内进行扩展,对 dirty memory 而言,这是真正的节省内存。

3、要用官方 API

现在,很多从类中获取数据的代码,必须同时处理那些有扩展数据和没有扩展数据的类,当然运行时会为你处理这一切,并且从外部看,一切都像往常一样工作,只是使用更少的内存。之所以会这样,是因为读取这些结构的代码,都在运行时内并且还会同时进行更新。坚持使用这些 API 真的很重要,因为任何试图直接访问这些数据结构的代码都将在今年的 OS 版本中停止工作,因为东西已经发生了改变,而且该代码不知道新的布局。我们看到,一些真实的代码由于这些变化而崩溃,除了你自己的代码之外,还要注意那些外部依赖性,你可能正把他们带入到你的 app 中,它们可能会在你没有意识到的情况下挖掘这些数据结构。这些结构中的所有信息都可通过官方API获得。

有一些函数,如 class_getNameclass_getSuperclass,当你使用这些 API 访问信息时,你知道,无论我们在后台进行什么更改,他们都将继续工作。所有的 API 都可以在 Objective-C 运行时说明文档中找到
class_getName
class_getSuperclass
class_copyMethodList
而这个文档在 developer.apple.com 中。

接下来,让我们更深入地了解一下这些类的数据结构,并看看另一个变化,Relative method lists 相对方法列表。
下一章我会继续总结,敬请期待!

以上的我写的内容是根据:2020 WWDC 大会视频 总结记录的,想看原视频的可以前去观看。

转载请备注原文出处,不得用于商业传播——凡几多

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,088评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,715评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,361评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,099评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 60,987评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,063评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,486评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,175评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,440评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,518评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,305评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,190评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,550评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,880评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,152评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,451评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,637评论 2 335

推荐阅读更多精彩内容