Android啟動優(yōu)化最佳方案:去啟動頁和異步初始化

項目地址:https://github.com/smartzheng/asyncstarter

隨著APP的日漸增大,集成的三方庫也越來越多,導致APP的啟動極其緩慢。最近在慕課get了一些不錯的優(yōu)化方案,將原來的冷啟動時間大概提升30%。
啟動的時間監(jiān)測可以直接用adb命令實現(xiàn):

adb shell am start -W PackageName/ActivityName

下面是我未優(yōu)化之前的項目debug版本啟動時間(華為p10plus),這里介紹一下幾個概念
ThisTime:最后一個Activity啟動耗時
TotalTime:所有Activity啟動耗時
WaitTimeTime:AMS啟動Activity啟動耗時
可以看到耗時接近1.3s多(1.3s不算長,但是往往應用加固之后還會慢大一截)。

-> ~ adb shell am start -W com.smartzheng/com.smartzheng.activity.SplashActivity
Starting: Intent { act=android.intent.action.MAIN cat=[android.intent.category.
LAUNCHER]cmp=com.smartzheng/.activity.MainActivity }
Warning: Activity not started, its current task has been brought to the front
Status: ok
Activity: com.uoko.mlgb/.mvp.view.activity.MainActivity
ThisTime: 600
TotalTime: 1301
WaitTime: 1330
Complete

優(yōu)化一:去掉啟動頁
IPC是個比較耗時的操作,往往我們會設置一個閃屏頁,去掉之后可以一定幅度減少啟動時間。
我的做法是直接刪除SplashActivity,將MainActivity設為啟動頁。然后在manifests中將其theme設為啟動時的theme:

<activity
    android:name=".activity.MainActivity"
    android:configChanges="screenSize|keyboardHidden|orientation"
    android:launchMode="singleTask"
    android:screenOrientation="portrait"
    android:theme="@style/SplashTheme">
    <intent-filter>
        <action android:name="android.intent.action.MAIN"/>
    </intent-filter>
    <intent-filter>
        <data android:scheme="growing.a4ce2f9edf74350a"/>
        <action android:name="android.intent.action.VIEW"/>
        <category android:name="android.intent.category.DEFAULT"/>
        <category android:name="android.intent.category.BROWSABLE"/>
    </intent-filter>
</activity>

SplashTheme中windowBackground為啟動圖片(提一句:這里的windowBackground最好是9patch圖片,防止拉伸變形),這樣設置還有一個好處是可以解決掉APP啟動白屏的問題。

<style name="SplashTheme" parent="AppThemeFullScreen">
    <item name="android:windowBackground">@drawable/splash</item>
    <item name="android:windowTranslucentNavigation">true</item>
</style>

最后,在MainActivity的onCreate方法中,手動setTheme更改為Application的默認Theme:

override fun onCreate(savedInstanceState: Bundle?) {
    setTheme(R.style.AppTheme)
    super.onCreate(savedInstanceState)
}

評論中有人問到如果去掉閃屏頁,那廣告位怎么解決?其實廣告頁面的顯示其實用一個單獨的Activity實現(xiàn)不是一個好選擇,直接調用Activity的addContentView方法就可以添加一個view在activity上面,在倒計時完成之后再removeContentView,這樣隨時都可以在任意activity上添加廣告,通用性和性能都更好。

優(yōu)化二:異步初始化
優(yōu)化之前,我的Application的onCreate是這樣的:

override fun onCreate() {
    super.onCreate()
    installSysExceptionHandler()
    initImPush()
    initGrowing()
    initBaidu()
    initApollo()
    initBugly()
    initIMSdk()
    initPush()
    initRxJava()
    initNineGridView()
    initSeoReport()
    initStrictMode()
    initFonts()
}

每一項初始化任務都是同步執(zhí)行的。優(yōu)化的思想就是將各個初始化任務放到一個線程池中去執(zhí)行。優(yōu)化之后代碼如下:

override fun onCreate() {
    super.onCreate()
    TaskDispatcher.init(this)
    TaskDispatcher.createInstance().run {
        addTask(InitApolloTask())
            .addTask(InitBaiduTask())
            .addTask(InitBuglyTask())
            .addTask(InitFontTask())
            .addTask(InitGrowingTask())
            .addTask(InitIMPushTask())
            .addTask(InitIMSDKTask())
            .addTask(InitJPushTask())
            .addTask(InitViewTask())
            .addTask(InitRxjavaTask())
       start()
       await()
    }
}

看看優(yōu)化之后的啟動耗時(release環(huán)境下),啟動速度大概提升了近30%:

-> ~ adb shell am start -W com.smartzheng/com.smartzheng.activity.SplashActivity
Starting: Intent { act=android.intent.action.MAIN cat=[android.intent.category.
LAUNCHER]cmp=com.smartzheng/.activity.MainActivity }
Status: ok
Activity: com.uoko.mlgb/.mvp.view.activity.MainActivity
ThisTime: 911
TotalTime: 911
WaitTime: 931
Complete

將各個初始化任務放到線程池執(zhí)行需要解決很多問題:
部分初始化工作只能在主線程執(zhí)行
部分初始化任務需要先執(zhí)行
部分初始化任務需要依賴其他任務執(zhí)行完之后才可以執(zhí)行
部分任務必須執(zhí)行完才能進入Activity頁面
參考課程,將封裝的代碼放到了GitHub,可以直接依賴使用;具體的實現(xiàn)細節(jié)在源碼中有詳細注釋。使用方法如下:
1.添加倉庫

allprojects {
    repositories {
        ...
        maven { url 'https://jitpack.io' }
    }
}
  1. 添加依賴
dependencies {
    implementation 'com.github.smartzheng:asyncstarter:1.0.1'
}

3.自定義Task

class InitTask : Task() {
    override fun needWait(): Boolean {//是否需要在阻塞在await(),在Application的onCreate方法之前執(zhí)行完
    return true
    }

    override fun dependsOn(): MutableList<Class<out Task>> {//等待另一個Task執(zhí)行完再執(zhí)行此任務初始化
        return mutableListOf(InitTask1::class.java)
    }
    override fun runOnMainThread(): Boolean {//是否需要運行在主線程
        return true
    }
    override fun needRunAsSoon(): Boolean {//提高優(yōu)先級,也可以指定優(yōu)先級大小priority
        return true
    }
    override fun run() {
        //初始化
    }
}

4.在Application中

class MyApplication : Application() {
    override fun onCreate() {
        super.onCreate()
        AsyncStarter.init(this)
        val starter = AsyncStarter.createInstance()

        starter.addTask(InitTask1())
            .addTask(InitTask2())
            .addTask(InitTask3())
            //addTask()...
        starter.start()
        starter.await()
        //Kotlin run
        //AsyncStarter.createInstance()
        //            .run {
        //                addTask(InitTask1())
        //                    .addTask(InitTask2())
        //                    .addTask(InitTask3())
        //                    //addTask()...
        //                    .start()
        //                await()
        //            }
    }
}

詳細的配置解釋和實例可以查看GitHub源碼,其中有詳細注釋,歡迎star:https://github.com/smartzheng/asyncstarter

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
【社區(qū)內容提示】社區(qū)部分內容疑似由AI輔助生成,瀏覽時請結合常識與多方信息審慎甄別。
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發(fā)布,文章內容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務。

相關閱讀更多精彩內容

友情鏈接更多精彩內容