項目啓動優化是每一個APP均可以接入的技術,只不過針對不一樣的業務邏輯咱們須要有不同的解決方案,由於有大部分人的「優化」,是在處理本身放蕩不羈的代碼。面試
既然這裏咱們要討論啓動優化,那麼咱們從啓動檢測開始。啓動檢測通常咱們會以main函數做爲分割點,main以前和main以後。
main以前稱爲per-main 階段。這個由dyld給你反饋應用的耗時。
main以後由開發者本身檢測。咱們能夠從main開始打點,到第一個頁面顯示爲止。bash
main函數以前的檢測蘋果提供了支持,具體配置方式來來來!上圖!微信
首先進入Edit Schemeapp
而後配置的 key 爲:DYLD_PRINT_STATISTICS函數
Total pre-main time: 123.34 milliseconds (100.0%) dylib loading time: 46.83 milliseconds (37.9%) rebase/binding time: 3.01 milliseconds (2.4%) ObjC setup time: 18.58 milliseconds (15.0%) initializer time: 54.91 milliseconds (44.5%) slowest intializers : libSystem.B.dylib : 5.00 milliseconds (4.0%) libBacktraceRecording.dylib : 9.43 milliseconds (7.6%) libMainThreadChecker.dylib : 17.72 milliseconds (14.3%) libViewDebuggerSupport.dylib : 20.72 milliseconds (16.8%)
載入動態庫,這個過程當中,會去裝載app使用的動態庫,而動態庫之間有它本身的依賴關係,因此會消耗時間去查找和讀取。
系統的動態庫,作了優化。因此從效率的角度來講,儘量使用系統庫。
而對於開發者定義導入的動態庫(dynamically linked shared library),則須要在花費更多的時間。Apple官方建議儘可能少的使用自定義的動態庫,或者考慮合併多個動態庫,其中一個建議是當大於6個的時候,則須要考慮合併它們。
在性能上出發將動態庫編譯成靜態庫也會優化這部分時間。工具
Rebase:在鏡像(MachO文件)內部調整指針的指向,針對mach-o在加載到內存中不是固定的首地址(ASLR)這一現象作數據修正的過程。
iOS4.3後引入了 ASLR ,MachO會被加載到隨機地址,這個隨機的地址跟代碼和數據指向的舊地址會有誤差。dyld 須要修正這個誤差,作法就是將 dylib 內部的指針地址都加上這個偏移量。
binding:將指針指向鏡像(MachO文件)外部的內容,binding就是將這個二進制調用的外部符號進行綁定的過程。性能
主要作如下幾件事來完成Objc Setup:
一、讀取二進制文件的 DATA 段內容,找到與 objc 相關的信息
二、註冊 Objc 類,ObjC Runtime 須要維護一張映射類名與類的全局表。當加載一個 MachO 時,它定義的全部的類都須要被註冊到這個全局表中;
三、讀取 protocol 以及 category 的信息,把category的定義插入方法列表 (category registration),
這一步的優化。學習
那麼針對這上面三個步驟,咱們能夠優化的方案是,不刻意的去減小几個類,可是能夠避免浪費。
隨着項目的不斷迭代,不少模塊和方法已經被廢棄可是卻一直留存在項目中,致使項目愈來愈臃腫。
咱們可使用一些工具來查找項目中沒有被用到的文件。從而達到優化。優化
一、Objc的+load()函數
二、C++的構造函數屬性函數 形如attribute((constructor)) void DoSomeInitializationWork()
咱們能作的就是將沒必要須在+load方法中作的事情延遲到+initialize中。
這是由於+load方法是在app啓動的時候就被調用,而+initialize方法則是在Class第一次使用的時候才調用,至關因而懶加載了。能夠把+load中的代碼移到initialize中,並結合dispatch_once來防止重複調用。
可是咱們項目中只有在使用method swizzling的時候會在+load中調用方法。因此這一點也沒什麼好優化的。spa
針對main函數以後的時間檢測就經過打點記錄。
在main()、didFinishLaunchingWithOptions以及第一個頁面的viewDidAppear中打點,進行記錄,從而來計算Main函數以後的時間。
文章的最後我想要補充一個小東西。就是三方應用的調試。由於既然咱們能夠檢測到性能,那麼看看別人的應用性能如何是一件不錯的事情。廢話很少說!
首先建立一個空工程運行到真機上!(咱們要利用這個工程的描述文件)
而後咱們須要一個腳本,腳本代碼以下:
# ${SRCROOT} 它是工程文件所在的目錄 TEMP_PATH="${SRCROOT}/Temp" #資源文件夾,咱們提早在工程目錄下新建一個APP文件夾,裏面放ipa包 ASSETS_PATH="${SRCROOT}/APP" #目標ipa包路徑 TARGET_IPA_PATH="${ASSETS_PATH}/*.ipa" #清空Temp文件夾 rm -rf "${SRCROOT}/Temp" mkdir -p "${SRCROOT}/Temp" #---------------------------------------- # 1. 解壓IPA到Temp下 unzip -oqq "$TARGET_IPA_PATH" -d "$TEMP_PATH" # 拿到解壓的臨時的APP的路徑 TEMP_APP_PATH=$(set -- "$TEMP_PATH/Payload/"*.app;echo "$1") # echo "路徑是:$TEMP_APP_PATH" #---------------------------------------- # 2. 將解壓出來的.app拷貝進入工程下 # BUILT_PRODUCTS_DIR 工程生成的APP包的路徑 # TARGET_NAME target名稱 TARGET_APP_PATH="$BUILT_PRODUCTS_DIR/$TARGET_NAME.app" echo "app路徑:$TARGET_APP_PATH" rm -rf "$TARGET_APP_PATH" mkdir -p "$TARGET_APP_PATH" cp -rf "$TEMP_APP_PATH/" "$TARGET_APP_PATH" #---------------------------------------- # 3. 刪除extension和WatchAPP.我的證書無法簽名Extention rm -rf "$TARGET_APP_PATH/PlugIns" rm -rf "$TARGET_APP_PATH/Watch" #---------------------------------------- # 4. 更新info.plist文件 CFBundleIdentifier # 設置:"Set : KEY Value" "目標文件路徑" /usr/libexec/PlistBuddy -c "Set :CFBundleIdentifier $PRODUCT_BUNDLE_IDENTIFIER" "$TARGET_APP_PATH/Info.plist" #---------------------------------------- # 5. 給MachO文件上執行權限 # 拿到MachO文件的路徑 APP_BINARY=`plutil -convert xml1 -o - $TARGET_APP_PATH/Info.plist|grep -A1 Exec|tail -n1|cut -f2 -d\>|cut -f1 -d\<` #上可執行權限 chmod +x "$TARGET_APP_PATH/$APP_BINARY" #---------------------------------------- # 6. 重簽名第三方 FrameWorks TARGET_APP_FRAMEWORKS_PATH="$TARGET_APP_PATH/Frameworks" if [ -d "$TARGET_APP_FRAMEWORKS_PATH" ]; then for FRAMEWORK in "$TARGET_APP_FRAMEWORKS_PATH/"* do #簽名 /usr/bin/codesign --force --sign "$EXPANDED_CODE_SIGN_IDENTITY" "$FRAMEWORK" done fi #注入 #yololib "$TARGET_APP_PATH/$APP_BINARY" "Frameworks/HankHook.framework/HankHook"
這個腳本能夠生成一個腳本文件,或者直接拷貝代碼去配置!腳本中每一個功能都寫上了註釋,不要無腦粘貼。
在工程根目錄下新建文件夾,並將脫殼的三方iPA包放進去(咱們已微信爲例)
給工程添加腳本
配置腳本並執行
接下來就能夠將三方的應用運行到真機了。下面是微信的啓動時間。pre-main大概1.1秒。
若是你想一塊兒進階,不妨添加一下交流羣1012951431
面試題資料或者相關學習資料都在羣文件中 進羣便可下載!