1、整体过程
(1)解析Info.plist
(2)Mach-O(可执行文件)加载
(3)程序执行
2、主要阶段:
分为两个阶段,pre-main阶段和main()阶段。程序启动到main函数执行前是pre-main阶段;在执行main函数后,调用AppDelegate中的-application:didFinishLaunchingWithOptions:
方法完成初始化,并展示首页,这是main()阶段,或者叫做main()之后阶段。
rebase
指针调整和bind
符号绑定。ObjC
的runtime
初始化(ObjC setup):ObjC
相关Class
的注册、category
注册、selector
唯一性检查等。+load()
方法、用attribute((constructor))
修饰的函数的调用、创建C++
静态全局变量等。1、pre-main阶段
对于pre-main阶段,Apple提供了一种测量方法,在 Xcode 中 Edit scheme -> Run -> Auguments 将环境变量DYLD_PRINT_STATISTICS 设为1 。之后控制台会输出类似内容,我们可以清晰的看到每个耗时:
从上面可以看出时间区域主要分为下面几个部分:
ASLR(Address Space Layout Randomization),地址空间布局随机化。在ASLR技术出现之前,程序都是在固定的地址加载的,这样hacker可以知道程序里面某个函数的具体地址,植入某些恶意代码,修改函数的地址等,带来了很多的危险性。ASLR就是为了解决这个的,程序每次启动后地址都会随机变化,这样程序里所有的代码地址都需要需要重新对进行计算修复才能正常访问。rebasing这一步主要就是调整镜像内部指针的指向。
Binding:将指针指向镜像外部的内容。
objc_init
方法,这个是runtime的初始化方法,在这个方法里面主要的操作就是加载类(对需要的class和category进行注册);load_images
方法,这个方法里面就是加载对应image里面的类,并调用load
方法(在下一阶段initializer)。load
方法,然后调用子类的,但是在load
里面不能调用[super load]
。最后才是调用category的load
方法。总之,所有的load
都会被调用到(注意:子类的initialize方法会覆盖父类,不同于load方法)。承接上一过程进行初始化(load)。如果我们代码里面使用了clang的__attribute__((constructor))
构造方法,这里会调用到。
2、main()阶段
测量main()函数开始执行到didFinishLaunchingWithOptions执行结束的时间,简单的方法:直接插入代码。(也可以使用其他工具)
三、改善APP的启动
建议应用的启动时间控制在400ms之下,并且在20s内启动,否则系统会kill app。优化APP的启动时间,需要就是分别优化pre-main和main的时间。
1、改善启动时pre-main阶段
(1)加载 Dylib
大部分ObjC初始化工作已经在Rebase/Bind阶段做完了,这一步dyld会注册所有声明过的ObjC类,将分类插入到类的方法列表里,再检查每个selector的唯一性。
在这一步倒没什么优化可做的,Rebase/Bind阶段优化好了,这一步的耗时也会减少。
(4)Initializers
到了这一阶段,dyld开始运行程序的初始化函数,调用每个Objc类和分类的+load方法,调用C/C++ 中的构造器函数(用attribute((constructor))修饰的函数),和创建非基本类型的C++静态全局变量。Initializers阶段执行完后,dyld开始调用main()函数。
在这一步,我们可以做的优化有:
(1)核心点:didFinishLaunchingWithOptions方法
这一阶段的优化主要是减少didFinishLaunchingWithOptions方法里的工作,在didFinishLaunchingWithOptions方法里我们经常会进行:
由于历史原因,这里的代码容易变得比较庞大,启动耗时难以控制。
(2)优化点:
满足业务需要的前提下,didFinishLaunchingWithOptions在主线程里做的事情越少越好。在这一步,我们可以做的优化有:
(1)+load
方法是一定会在runtime中被调用的。只要类被添加到runtime中了,就会调用+load
方法,即只要是在Compile Sources
中出现的文件总是会被装载,与这个类是否被用到无关,因此+load
方法总是在main函数之前调用。
(2)+load
方法不会覆盖。也就是说,如果子类实现了+load
方法,那么会先调用父类的+load
方法(无需手动调用super),然后又去执行子类的+load
方法。
(3)+load方法只会调用一次。
(4)+load方法执行顺序是:类 -> 子类 ->分类。而不同分类之间的执行顺序不一定,依据在Compile Sources
中出现的顺序(先编译,则先调用,列表中在下方的为“先”)。
(1)+initialize
方法是在类或它的子类收到第一条消息之前被调用的,这里所指的消息包括实例方法和类方法的调用。因此+initialize
方法总是在main函数之后调用。
(2)+initialize
方法只会调用一次。
(3)+initialize
方法实际上是一种惰性调用,如果一个类一直没被用到,那它的+initialize
方法也不会被调用,这一点有利于节约资源。
(4)+initialize
方法会覆盖。如果子类实现了+initialize
方法,就不会执行父类的了,直接执行子类本身的。如果分类实现了+initialize
方法,也不会再执行主类的。
(5)+initialize
方法的执行覆盖顺序是:分类 -> 子类 ->类。且只会有一个+initialize
方法被执行。
(6)+initialize
方法是发送消息(objc_msgSend()),如果子类没有实现+initialize
方法,也会自动调用其父类的+initialize
方法。
+load
方法在main函数之前执行,+initialize
在main函数之后执行。+load
方法的话,子类不会调用父类的+load
方法;而子类如果没有实现+initialize
方法的话,也会自动调用父类的+initialize
方法。+load
方法是在类被装在进来的时候就会调用,+initialize
在第一次给某个类发送消息时调用(比如实例化一个对象),并且只会调用一次,是懒加载模式,如果这个类一直没有使用,就不回调用到+initialize
方法。
(1)+load
一般是用来交换方法Method Swizzle
,由于它是线程安全的,而且一定会调用且只会调用一次,通常在使用UrlRouter的时候注册类的时候也在+load
方法中注册。(2)+initialize
方法主要用来对一些不方便在编译期初始化的对象进行赋值,或者说对一些静态常量进行初始化操作。
原文:https://www.cnblogs.com/feng9exe/p/15091183.html