APP啟動(dòng)
APP的啟動(dòng)方式
- 冷啟動(dòng)(Cold Launch):從零開始啟動(dòng)APP
- 熱啟動(dòng)(Warm Launch):APP已經(jīng)在內(nèi)存中,掛在后臺(tái),再次點(diǎn)擊APP圖標(biāo)啟動(dòng)APP
APP冷啟動(dòng)(可以概括為3/2大階段)
- dyld
- runtime
- main
或者 - main 函數(shù)執(zhí)行前
- main 函數(shù)執(zhí)行后
先說3階段
APP冷啟動(dòng) - dyld
dyld(dynamic link editor)贝咙,Apple的動(dòng)態(tài)鏈接器涉兽,可以用來裝載Mach-O文件(可執(zhí)行文件瓶您、動(dòng)態(tài)庫(kù)等)
dyld所做的事情有:
- 裝載APP的可執(zhí)行文件胸懈,同時(shí)會(huì)遞歸加載所有依賴的動(dòng)態(tài)庫(kù)
- 當(dāng)dyld把可執(zhí)行文件啦粹、動(dòng)態(tài)庫(kù)都裝載完畢后榜掌,會(huì)通知Runtime進(jìn)行下一步的處理
APP冷啟動(dòng) - runtime
runtime所做的事情有:
- 調(diào)用map_images進(jìn)行可執(zhí)行文件內(nèi)容的解析和處理
- 在load_images中調(diào)用call_load_methods缴罗,調(diào)用所有Class和Category的+load方法
- 進(jìn)行各種objc結(jié)構(gòu)的初始化(注冊(cè)O(shè)bjc類 、初始化類對(duì)象等等)
- 調(diào)用C++靜態(tài)初始化器和attribute((constructor))修飾的函數(shù)
- 到此為止烙丛,可執(zhí)行文件和動(dòng)態(tài)庫(kù)中所有的符號(hào)(Class舅巷,Protocol,Selector河咽,IMP钠右,…)都已經(jīng)按格式成功加載到內(nèi)存中,被runtime 所管理
APP冷啟動(dòng) - main函數(shù)執(zhí)行
請(qǐng)看下面 main 函數(shù)執(zhí)行后:
2階段
main 函數(shù)執(zhí)行前:
- 首先當(dāng)程序啟動(dòng)時(shí)忘蟹,系統(tǒng)會(huì)讀取程序的可執(zhí)行文件(mach-o), 從里面獲取動(dòng)態(tài)加載器(dylb)的路徑;
- 加載dylb, dylb會(huì)初始化運(yùn)行環(huán)境飒房,配合ImageLoader將二進(jìn)制文件加載到內(nèi)存中去;
- 動(dòng)態(tài)鏈接依賴庫(kù), 初始化依賴庫(kù),初始化 runtime;
- runtime 會(huì)對(duì)項(xiàng)目中的所有類進(jìn)行類結(jié)構(gòu)初始化媚值,調(diào)用所有的 load 方法;
- 最后 dylb 會(huì)返回 main 函數(shù)地址情屹,main 函數(shù)被調(diào)用,進(jìn)入程序入口
main 函數(shù)執(zhí)行后:
- 內(nèi)部會(huì)調(diào)用 UIApplicationMain 函數(shù)杂腰,創(chuàng)建一個(gè)UIApplication對(duì)象和它的代理垃你,就是我們項(xiàng)目中的 Appdelegate 類
- 開啟一個(gè)事件循環(huán)(main runloop), 監(jiān)聽系統(tǒng)事件
- 程序啟動(dòng)完畢時(shí),通知代理Appdelegate, 調(diào)用 didFinishLaunching 代理方法喂很,在這里會(huì)創(chuàng)建 UIWindow,設(shè)置它的rootViewController,
- 最后調(diào)用 self.window makeKeyAndVisable顯示窗口
小結(jié)
- APP的啟動(dòng)由dyld主導(dǎo)惜颇,將可執(zhí)行文件加載到內(nèi)存,順便加載所有依賴的動(dòng)態(tài)庫(kù)
- 并由runtime負(fù)責(zé)加載成objc定義的結(jié)構(gòu)
- 所有初始化工作結(jié)束后少辣,dyld就會(huì)調(diào)用main函數(shù)
- 接下來就是UIApplicationMain函數(shù)凌摄,AppDelegate的application:didFinishLaunchingWithOptions:方法
APP的啟動(dòng)優(yōu)化
dyld
- 減少動(dòng)態(tài)庫(kù)、合并一些動(dòng)態(tài)庫(kù)(定期清理不必要的動(dòng)態(tài)庫(kù))
- 減少Objc類漓帅、分類的數(shù)量锨亏、減少Selector數(shù)量(定期清理不必要的類、分類)
- 減少C++虛函數(shù)數(shù)量
- Swift盡量使用struct
runtime
- 用+initialize方法和dispatch_once取代所有的attribute((constructor))忙干、C++靜態(tài)構(gòu)造器器予、ObjC的+load
main
- 在不影響用戶體驗(yàn)的前提下,盡可能將一些操作延遲捐迫,不要全部都放在finishLaunching方法中(例:廣告頁(yè)異步加載緩存本地,等待下次啟動(dòng)在進(jìn)行顯示)
- 按需加載