[TOC]
參加官方文檔:[https://developer.android.com/topic/performance/vitals/launch-time?hl=zh-cn#top\_of\_page](https://developer.android.com/topic/performance/vitals/launch-time?hl=zh-cn#top_of_page)
# 應用啟動內部機制
應用有三種啟動狀態,冷啟動中應用從頭開始啟動,其他狀態下系統需要將后臺運行的應用帶入前臺,我們在冷啟動的基礎上進行優化,可以最大限度的提升應用性能
## 冷啟動
出現場景:
* 設備剛開機,啟動應用時
* 系統終止應用進程(或用戶手動劃掉應用),再次重新打開時
冷啟動分為三步:
1、加載并啟動應用
2、在啟動后立即顯示應用的空白啟動窗口
3、創建應用進程。系統首次完成應用繪制后,會換掉應用的空白啟動窗口,展示具體的UI

## 熱啟動
出現場景:
* 用戶直接點擊Home鍵回到主頁或最近應用鍵到其他應用,再次返回本應用時
熱啟動中,系統的所有工作就是將后臺的 Activity 帶到前臺。如果應用的所有 Activity 都還駐留在內存中,則應用可以無須重復對象初始化、布局擴充和呈現。熱啟動顯示的屏幕上行為和冷啟動場景相同,即系統進程顯示空白屏幕,直到應用完成 Activity 呈現。
## 溫啟動
出現場景:
* 用戶退出應用,但并未直接結束掉(劃掉)進程
* 應用被回收,重新打開時(onCreate中會拿到回收時保存的實例狀態)
溫啟動的開銷比熱啟動多,用戶退出應用,之后又重新啟動,進程可能已經繼續運行,但是Activity需要重新創建。
# 檢測和診斷問題
應用啟動時間過長時,Google Play管理中心可協助提醒我們,此情況在國內環境中暫不使用,但其標準可以做個參考。Android Vitals 在應用出現以下情況時將其啟動時間視為過長:
* 冷啟動用了 5 秒或更長時間
* 溫啟動用了 2 秒或更長時間
* 熱啟動用了 1.5 秒或更長時間
## 初步顯示啟動所用時間
在 Android 4.4(API 級別 19)及更高版本中,logcat 包括一個輸出行,其中包含名為`Displayed`的值。此值代表從啟動進程到在屏幕上完成對應 Activity 繪制所經過的時間。經過的時間包括以下事件序列:
1. 啟動進程。
2. 初始化對象。
3. 創建并初始化 Activity。
4. 擴充布局。
5. 首次繪制應用。
報告的日志行類似于以下示例:
```plain
ActivityManager: Displayed com.android.myexample/.StartupTiming: +3s534ms
```
也可以使用 ADB Shell Activity Manager 命令運行應用來測量初步顯示所用時間。
## 完全顯示啟動所用時間
某些情況下Activity視圖會出現延遲加載的情況,比如界面中網絡請求獲取到的圖片就會因網絡環境不同而延遲加載,我們可以在Activity中手動調用`reportFullyDrawn()`方法,來測量從應用啟動到完全顯示所有資源和視圖層次結構所經過的時間。
# 問題和優化方案
## 常見影響啟動時間的問題
### 1、密集型應用初始化
即在Application的onCreate方法中執行了過多操作。
優化解決方案:
* 延遲初始化對象,僅初始化立即需要的對象
* 不要在Application中創建全局靜態對象,而是移動到單例模式中
* 考慮使用依賴注入框架(如Dagger),在首次注入時創建對象和依賴項
### 2、密集型Activity初始化
創建 Activity 通常需要進行大量的高開銷工作,通常有機會優化這項工作以實現性能改進。通常視圖層次結構越大,應用擴充它所花的時間就越長;并且在主線程初始化資源也可能會降低啟動速度。
優化解決方案:
* 減少冗余或嵌套布局,優化布局結構
* 不要擴充在啟動期間無需顯示的界面部分,考慮使用ViewStub對象在更合適的時機擴充
* 在子線程延遲部分資源的初始化
* 允許應用先加載并顯示您的視圖,稍后再更新依賴于位圖和其他資源的可視屬性
## 啟動白屏問題與優化
點擊應用圖標,打開App,創建應用進程時,系統會根據清單文件設置的主題顏色來展示一個白屏或黑屏,以給與用戶反饋。當創建應用進程這一步耗時較長時,白屏或黑屏體驗就很不好了,來看看相應的解決方案:
### 1、禁用系統進程繪制時的空白屏幕
將啟動Activity的主題設置為如下即可:
```xml
<style name="APPTheme" parent="@android:style/Theme.Holo.NoActionBar">
<item name="android:windowDisablePreview">true</item>
</style>
```
禁用系統進程繪制時的空白屏幕,可能導致啟動時間更長,最重要的是用戶點擊了圖標后,界面會無響應一段時間,沒有相應的反饋。
### 2、為啟動Activity自定義主題背景
```xml
<layer-list xmlns:android="http://schemas.android.com/apk/res/android" android:opacity="opaque">
<!-- The background color, preferably the same as your normal theme -->
<item android:drawable="@android:color/white"/>
<!-- Your product logo - 144dp color version of your app icon -->
<item>
<bitmap
android:src="@drawable/product_logo_144dp"
android:gravity="center"/>
</item>
</layer-list>
```
```xml
<style name="APPTheme" parent="@android:style/Theme.Holo.NoActionBar">
<item name="android:windowBackground">@drawable/splash_icon</item>
</style>
```
為啟動Activity設置以上主題,這樣自定義的主題背景會替代掉默認的白屏或黑屏。
- 導讀
- Java知識
- Java基本程序設計結構
- 【基礎知識】Java基礎
- 【源碼分析】Okio
- 【源碼分析】深入理解i++和++i
- 【專題分析】JVM與GC
- 【面試清單】Java基本程序設計結構
- 對象與類
- 【基礎知識】對象與類
- 【專題分析】Java類加載過程
- 【面試清單】對象與類
- 泛型
- 【基礎知識】泛型
- 【面試清單】泛型
- 集合
- 【基礎知識】集合
- 【源碼分析】SparseArray
- 【面試清單】集合
- 多線程
- 【基礎知識】多線程
- 【源碼分析】ThreadPoolExecutor源碼分析
- 【專題分析】volatile關鍵字
- 【面試清單】多線程
- Java新特性
- 【專題分析】Lambda表達式
- 【專題分析】注解
- 【面試清單】Java新特性
- Effective Java筆記
- Android知識
- Activity
- 【基礎知識】Activity
- 【專題分析】運行時權限
- 【專題分析】使用Intent打開三方應用
- 【源碼分析】Activity的工作過程
- 【面試清單】Activity
- 架構組件
- 【專題分析】MVC、MVP與MVVM
- 【專題分析】數據綁定
- 【面試清單】架構組件
- 界面
- 【專題分析】自定義View
- 【專題分析】ImageView的ScaleType屬性
- 【專題分析】ConstraintLayout 使用
- 【專題分析】搞懂點九圖
- 【專題分析】Adapter
- 【源碼分析】LayoutInflater
- 【源碼分析】ViewStub
- 【源碼分析】View三大流程
- 【源碼分析】觸摸事件分發機制
- 【源碼分析】按鍵事件分發機制
- 【源碼分析】Android窗口機制
- 【面試清單】界面
- 動畫和過渡
- 【基礎知識】動畫和過渡
- 【面試清單】動畫和過渡
- 圖片和圖形
- 【專題分析】圖片加載
- 【面試清單】圖片和圖形
- 后臺任務
- 應用數據和文件
- 基于網絡的內容
- 多線程與多進程
- 【基礎知識】多線程與多進程
- 【源碼分析】Handler
- 【源碼分析】AsyncTask
- 【專題分析】Service
- 【源碼分析】Parcelable
- 【專題分析】Binder
- 【源碼分析】Messenger
- 【面試清單】多線程與多進程
- 應用優化
- 【專題分析】布局優化
- 【專題分析】繪制優化
- 【專題分析】內存優化
- 【專題分析】啟動優化
- 【專題分析】電池優化
- 【專題分析】包大小優化
- 【面試清單】應用優化
- Android新特性
- 【專題分析】狀態欄、ActionBar和導航欄
- 【專題分析】應用圖標、通知欄適配
- 【專題分析】Android新版本重要變更
- 【專題分析】唯一標識符的最佳做法
- 開源庫源碼分析
- 【源碼分析】BaseRecyclerViewAdapterHelper
- 【源碼分析】ButterKnife
- 【源碼分析】Dagger2
- 【源碼分析】EventBus3(一)
- 【源碼分析】EventBus3(二)
- 【源碼分析】Glide
- 【源碼分析】OkHttp
- 【源碼分析】Retrofit
- 其他知識
- Flutter
- 原生開發與跨平臺開發
- 整體歸納
- 狀態及狀態管理
- 零碎知識點
- 添加Flutter到現有應用
- Git知識
- Git命令
- .gitignore文件
- 設計模式
- 創建型模式
- 結構型模式
- 行為型模式
- RxJava
- 基礎
- Linux知識
- 環境變量
- Linux命令
- ADB命令
- 算法
- 常見數據結構及實現
- 數組
- 排序算法
- 鏈表
- 二叉樹
- 棧和隊列
- 算法時間復雜度
- 常見算法思想
- 其他技術
- 正則表達式
- 編碼格式
- HTTP與HTTPS
- 【面試清單】其他知識
- 開發歸納
- Android零碎問題
- 其他零碎問題
- 開發思路