Android的冷啟動與熱啟動以及優化方案
熱啟動
定義:
當啟動應用時,後臺已有該應用的程序(例:按back鍵、home鍵,應用雖然會退出,但是該應用的程序是依然會保留在後臺,可進入任務列表檢視),所以在已有程序的情況下,這種啟動會從已有的程序中來啟動應用,這個方式叫熱啟動。
特點:
熱啟動因為會從已有的程序中來啟動,所以熱啟動就不會走Application這步了,而是直接走MainActivity(包括一系列的測量、佈局、繪製),所以熱啟動的過程只需要建立和初始化一個MainActivity就行了,而不必建立和初始化Application,
因為一個應用從新程序的建立到程序的銷燬,Application只會初始化一次。
冷啟動
定義:
當啟動應用時,後臺沒有該應用的程序,這時系統會重新建立一個新的程序分配給該應用,這個啟動方式就是冷啟動。
特點:
冷啟動因為系統會重新建立一個新的程序分配給它,所以會先建立和初始化Application類,再建立和初始化MainActivity類(包括一系列的測量、佈局、繪製),最後顯示在介面上。
windowBackground等屬性設定給MainActivity以及配置Activity層級上的一些屬性、再inflate佈局、當onCreate/onStart/onResume方法都走完了後最後才進行contentView的measure/layout/draw顯示在介面上,所以直到這裡,
應用的第一次啟動才算完成,這時候我們看到的介面也就是所說的第一幀。所以,總結一下,應用的啟動流程如下:
Application的構造器方法——>attachBaseContext()——>onCreate()——>Activity的構造方法——>onCreate()——>配置主題中背景等屬性——>onStart()——>onResume()——>測量佈局繪製顯示在介面上。
冷啟動實現APP秒開
冷啟動時間:
冷啟動時間是指使用者從手機桌面點選APP的那一刻起到啟動頁面的Activity呼叫onCreate()方法之間的這個時間段
冷啟動時間段內發生的事情:
首先我們要知道當開啟一個Activity的時候發生了什麼,在一個Activity開啟時,如果該Activity所屬的Application還沒有啟動,那麼系統會為這個Activity建立一個程序(每建立一個程序都會呼叫一次Application,所以Application的onCreate()方法可能會被呼叫多次),在程序的建立和初始化中,勢必會消耗一些時間,在這個時間裡,WindowManager會先載入APP裡的主題樣式裡的視窗背景(windowBackground)作為預覽元素,然後才去真正的載入佈局,如果這個時間過長,而預設的背景又是黑色或者白色,這樣會給使用者造成一種錯覺,這個APP很卡,很不流暢,自然也影響了使用者體驗。
優化方案
消除啟動時的白屏/黑屏
在使用者點選手機桌面APP的時候,看到的黑屏或者白屏其實是介面渲染前的第一幀,如果你看懂了文章頭的那2個問題,那麼解決這個問題就非常輕鬆了,無非就是將Theme裡的windowBackground設定成我們想要讓使用者看到的畫面就可以了,這裡有2種做法:
1、將背景圖設定成我們APP的Logo圖,作為APP啟動的引導,現在市面上大部分的APP也是這麼做的。
<style name="AppWelcome" parent="AppTheme">
<item name="android:windowBackground">@mipmap/bg_welcome_start</item>
</style>
方案1優化後
2、將背景顏色設定為透明色,這樣當用戶點選桌面APP圖片的時候,並不會"立即"進入APP,而且在桌面上停留一會,其實這時候APP已經是啟動的了,只是我們心機的把Theme裡的windowBackground的顏色設定成透明的,強行把鍋甩給了手機應用廠商(手機反應太慢了啦,哈哈),其實現在微信也是這樣做的,不信你可以試試。
<style name="Appwelcome" parent="android:Theme.Translucent.NoTitleBar.Fullscreen"/>
透明化這種做法需要注意的一點,如果直接把Theme引入Activity,在執行的時候可能會出現如下異常:
java.lang.IllegalStateException: You need to use a Theme.AppCompat theme (or descendant) with this activity.
這個是因為使用了不相容的Theme,例如我這裡的Activity繼承了AppCompatActivity,解決方案很簡單:
1、讓其Activity整合Activity而不要整合相容性的AppCompatActivity
2、在onCreate()方法裡的super.onCreate(savedInstanceState)之前設定我們原來APP的Theme
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
setTheme(R.style.AppTheme);
super.onCreate(savedInstanceState);
}
}
方案2優化後
上面的2種做法,我們都需要將Theme引入對應的Activity
<activity
android:name=".app.main.MainActivity"
android:theme="@style/AppWelcome"
android:screenOrientation="portrait">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>