天天看點

android冷啟動中(歡迎頁splash頁)中的白屏問題

1、冷啟動:當啟動應用時,背景沒有該應用的程序,這時系統會重新建立一個新的程序配置設定給該應用,這個啟動方式就是冷啟動。

特點:冷啟動因為系統會重新建立一個新的程序配置設定給它,是以會先建立和初始化application類,再建立和初始化MainActivity類(包括一系列的測量、布局、繪制),最後顯示在界面上。

2、熱啟動:當啟動應用時,背景已有該應用的程序(例:按back鍵、home鍵,應用雖然會退出,但是該應用的程序是依然會保留在背景,可進入任務清單檢視),是以在已有程序的情況下,這種啟動會從已有的程序中來啟動應用,這個方式叫熱啟動。

3、在冷啟動的時間段内發生了什麼?

首先我們要知道當打開一個Activity的時候發生了什麼,在一個Activity打開時,如果該Activity所屬的Application還沒有啟動,那麼系統會為這個Activity建立一個程序(每建立一個程序都會調用一次Application,是以Application的onCreate()方法可能會被調用多次),在程序的建立和初始化中,勢必會消耗一些時間,在這個時間裡,WindowManager會先加載APP裡的主題樣式裡的視窗背景(windowBackground)作為預覽元素,然後才去真正的加載布局,如果這個時間過長,而預設的背景又是黑色或者白色,這樣會給使用者造成一種錯覺,這個APP很卡,很不流暢,自然也影響了使用者體驗。

特點:熱啟動因為會從已有的程序中來啟動,是以熱啟動就不會走application這步了,而是直接走MainActivity(包括一系列的測量、布局、繪制),是以熱啟動的過程隻需要建立和初始化一個MainActivity就行了,而不必建立和初始化application,因為一個應用從新程序的建立到程序的銷毀,application隻會初始化一次。

既然上述問題不是出在application,那麼肯定就是在Activity了,是setContentView導緻的

解決方法:

1.我們可以定義一個theme,然後給theme設定背景就好了:

<style name="StartTheme" parent="AppTheme">
    <item name="android:windowBackground">@mipmap/icon_splash</item>
</style>
           

2.這麼做有個問題,圖檔的記憶體會釋放不掉,是以放在activity的super調用前,用流資源方式加載圖檔,設定到window的背景中去就好了

3.将背景顔色設定為透明色,這樣當使用者點選桌面APP圖檔的時候,并不會”立即”進入APP,而且在桌面上停留一會,其實這時候APP已經是啟動的了,隻是我們心機的把Theme裡的windowBackground的顔色設定成透明的,強行把鍋甩給了手機應用廠商(手機反應太慢了啦,哈哈),其實作在微信也是這樣做的,不信你可以試試。

透明化這種做法需要注意的一點,如果直接把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);
    }
}
           

上面的做法,我們也需要将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>