Android启动优化最佳方案:去启动页和异步初始化

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

随着APP的日渐增大,集成的三方库也越来越多,导致APP的启动极其缓慢。最近在慕课get了一些不错的优化方案,将原来的冷启动时间大概提升30%。
启动的时间监测可以直接用adb命令实现:

adb shell am start -W PackageName/ActivityName

下面是我未优化之前的项目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

优化一:去掉启动页
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实现不是一个好选择,直接调用Activity的addContentView方法就可以添加一个view在activity上面,在倒计时完成之后再removeContentView,这样随时都可以在任意activity上添加广告,通用性和性能都更好。

优化二:异步初始化
优化之前,我的Application的onCreate是这样的:

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

每一项初始化任务都是同步执行的。优化的思想就是将各个初始化任务放到一个线程池中去执行。优化之后代码如下:

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()
    }
}

看看优化之后的启动耗时(release环境下),启动速度大概提升了近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

将各个初始化任务放到线程池执行需要解决很多问题:
部分初始化工作只能在主线程执行
部分初始化任务需要先执行
部分初始化任务需要依赖其他任务执行完之后才可以执行
部分任务必须执行完才能进入Activity页面
参考课程,将封装的代码放到了GitHub,可以直接依赖使用;具体的实现细节在源码中有详细注释。使用方法如下:
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方法之前执行完
    return true
    }

    override fun dependsOn(): MutableList<Class<out Task>> {//等待另一个Task执行完再执行此任务初始化
        return mutableListOf(InitTask1::class.java)
    }
    override fun runOnMainThread(): Boolean {//是否需要运行在主线程
        return true
    }
    override fun needRunAsSoon(): Boolean {//提高优先级,也可以指定优先级大小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

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,185评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,445评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,684评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,564评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,681评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,874评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,025评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,761评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,217评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,545评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,694评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,351评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,988评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,778评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,007评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,427评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,580评论 2 349

推荐阅读更多精彩内容