iOS性能优化--启动优化(一)

测量启动过程的耗时

当按下home键的时候,App进程并不会马上被干掉,还会在后台存活一定时间。在这个时间内如果再次回到App那么几乎不需要做什么,就可以还原到退出前的状态。这种持续存活的情况下启动App,我们称为热启动,相对而言冷启动就是App被kill掉以后一切从头开始启动的过程。

测量main()函数之前的启动时间

苹果已经提供了这个测量方式
在Xcode的菜单中选择Project→Scheme→Edit Scheme,然后找到 Run → Environment Variables,为项目添加环境变量DYLD_PRINT_STATISTICS 值为YES

这样Xcode在启动App后会在控制台输出启动耗时

main()函数之后

main()函数开始至applicationWillFinishLaunching结束,我们统一称为main()函数之后的部分。

虚拟内存和物理内存

早期计算机没有虚拟内存这个问题,所有的地址都是实实在在的物理地址。当应用加载到内存中时,是全部加载到内中。内存的访问都是直接访问物理内存地址,这样极其不安全。然后就是很多时候一个应用真正只是用到少量的内存,这样就存在内存的浪费。因此出现了虚拟内存,让每个应用在逻辑上存在一大片连续的虚拟内存,每个进程的虚拟内存对应一个映射表,映射到实际的物理内存上。
那么cpu在访问这个进程的时候,先通过虚拟地址寻址,然后转换为对应的物理地址(地址翻译,这需要对应的硬件--cpu的内存管理单元mmu与操作系统配合)。
也就是说,进程中使用的地址是一片单独连续的虚拟地址,通过进程映射表(页表)映射到物理内存中,这时候进程在物理内存上占用的空间不一定是连续的。这样就解决了安全和物理内存使用率的问题。

内存分页

进程有自己的虚拟内存,但是物理内存实际也没有增大,这时候要解决内存的使用效率。
进程自己对应的映射表是以页为单位,在macOS上是以4KB一页为单位,iOS是以16KB为单位。(终端输入 $PAGESIZE 可以查看到macOS的分页大小)。
如何解决内存浪费的?
应用程序加载到内存中时,并不会全部加载到物理内存中,属于懒加载,用哪一部分就加载那一部分。当访问进程的内存地址时,首先看页表,查看所要访问的对应页表是否已经加载到内存中。如果这一页没有在物理内存中时,操作系统会阻塞当前进程,发出一个缺页异常/缺页中断(pagefault),让后将磁盘中对应页的数据加载到内存中,完成虚拟内存和物理内存的映射。
当前进程的页表数据加载到物理内存中时,不一定是连续的,也有可能会覆盖其他进程的不活跃页,这样的按需分配,极大提高内存的使用效率。

虚拟内存的安全问题

虚拟内存通过页表映射到物理内存上,因此直接访问物理地址并不能实际正确的拿到进程的数据,但是进程的虚拟内存地址相对于自己来说也是绝对的,不管程序运行多少次,如果访问同一个函数,它在虚拟内存中的地址都是一样的这样也存在安全问题(比如直接静态注入)。
这样也出现了新的技术--ASLR(Address Space Layout Randomization)。
每次虚拟内存在加载之前,都加一个随机偏移值。iOS好像是从4.3版本开始了。

二进制重排

缺页中断/缺页异常:内存分页管理,每一页加载的时候都会发生。
在iOS中,在加载缺页内存的时候,不仅发生缺页阻塞从磁盘中加载数据,还要对加载的这页做签名的验证
在使用中,我们一般感受不到这个过程,但是在启动中,这个过程也许你能很好的体会过。启动时,程序有大量的代码需要加载、执行,那么这个缺页中断有可能就很明显了。

如何优化?
假如我的app需要加载10页内存,但是启动的时候需要加载的代码放在1、3、5页。这时候来看看,代码在mac-o文件中的位置是根据文件加载生成的顺序来决定。那么这时候app启动需要运行的代码放在3个虚拟内存页中就需要出现3次pagefault。
如果我们那需要启动时用的代码全部放在前面1-2页中,甚至如果代码足够小只需要1页就够了,这样极大减少进程的阻塞。这也就是二进制重排。

查看pagefault

Xcode提供相关的调试工具,使用自带的instruments查看MainThread中虚拟内存的file backed page in项目,它代表着启动时,产生的pagefault次数。
二进制重排的优化是发生在编译链接阶段,对即将生成的二进制可执行文件进行重排。
xcode使用的连接器叫ld它可以指向一个order_file文件,在这个文件中指定排列符号,那么Xcode在编译时会按照指定的排列编译出可执行的文件,苹果自己也是这么用的。

开始做优化

首先看看我们项目的link map文件,在项目的build settings中开启这个文件的输出


然后重新编译一下,然后可以在工程的build目录里面找到一份link map文件

这个文件里面就记录一些链接.o的文件、mac-o文件里的一些信息、符号信息symbols等等…

注意,这个symbols就是关注的要点:默认情况下 它是按照文件排列顺序链接了。

image.png

通过order文件重新排列加载顺序:
在工程配置中,添加一个指定符号顺序的order文件后,让编译器按照指定的顺序重新排列二进制文件,把最需要加载的代码段放在内存页靠前的位置。

这里只是演示了让viewcontroller中的几个自定义方法优先靠排列在内存分页中,实际中一个app启动时的pagefault可能多达几千次,那么需要重排的函数远不止这一点。

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