声明:本篇文章已授权微信公众号 YYGeeker 独家发布。
对于一个APP来说,启动秒开,切换顺畅的体验能给用户留下良好的第一印象,启动速度对于用户体验及提高用户留存的重要性不言而喻。那么我们首先从它开始入手,从理论结合实际来谈谈有哪些优化启动速度及性能的技巧。
Google 官方介绍文档:https://developer.android.com/topic/performance/vitals/launch-time 有兴趣可以自行阅读。
Google 对应用的启动定义了三个概念,分为冷启动、热启动、温启动。而启动最耗时同时也是我们主要去优化的地方,就是冷启动。在冷启动App之时,手机系统会先执行以下三个任务:
这三个任务执行完毕之后,我们的App进程就创建成功了,然后会执行以下操作:
从系统层面来看,一个 Activity 走完 onCreate/onStart/onResume 这几个生命周期之后,只是完成了应用自身的一些配置,比如 window 的一些属性的设置/View树的建立,并没有显示。换句话来说,其实到这一步系统只是调用了 inflate 而已。后面 ViewRootImpl 还会调用两次performTraversals ,初始化 Egl 以及 measure/layout/draw 等。
因此,在Android系统里,我们定义一个应用的启动时间, 肯定不能以Activity 的回调函数作为基准,而应该以用户在手机屏幕上看到我们在 onCreate 的 setContentView 中设置的 layout 完全显示为准,也就是我们常说的应用第一帧。而一旦成了第一次绘制,系统进程就会用Main Activity替换掉之前已经展示的Background Window。
App进程的创建等环节,我们无法去主动干涉控制。那么我们可以优化启动速度的方向有哪些呢? 本篇文章将围绕它展开讨论。
Google对启动时长定义了这三个概念:
ThisTime、TotalTime 的值在 frameworks\base\services\core\java\com\android\server\am\ActivityRecord.java 文件的 reportLaunchTimeLocked() 函数中计算得到,有兴趣可自行翻阅。
一般来说,开发者只要关心TotalTime即可,这个时间才是我们自己应用真正启动的耗时。一般来说我们可通过如下几种方法检测启动耗时:
在Android 4.4(API级别19)及更高版本中,系统会输出一个包含名为Displayed的值的输出行。此值表示Activity启动过程和完成在屏幕上绘制相应活动之间所经过的时间长度。
我们在Android Studio的Logcat可以查看这个输出信息,需要注意的是我们在logcat视图中,需要去除过滤器,选择 No Filters。因为系统的输出信息是在系统进程服务,而不是应用程序本身输出的启动日志,具体可参考下图:
通过adb启动我们的Activity或Service,控制台会输出应用的启动时间,cmd命令格式如下:
adb shell am start -W [packagename/activity]
如执行 手机YY 启动时间统计命令:
adb shell am start -W com.duowan.mobile/com.yy.mobile.ui.splash.SplashActivity
log 打印如下:
Starting: Intent { act=android.intent.action.MAIN cat=[android.intent.category.LAUNCHER] cmp=com.duowan.mobile/com.yy.mobile.ui.splash.SplashActivity }
Status: ok
Activity: com.duowan.mobile/com.yy.mobile.ui.splash.SplashActivity
ThisTime: 1794
TotalTime: 1794
WaitTime: 1831
Complete
通过Trace文件我们也可以分析启动时的相关信息,在ANR触发时,系统会自动生成Trace文件以供我们去分析。一般来说,正常情况下Trace 文件可以有以下几种主动生成的方式:
代码生成
API 19或者以上可以通过如下方式打印trace 文件:
Debug.startMethodTracing("test"); //开始 trace(保存文件到 "/sdcard/test.trace" )
// 省略业务代码...
Debug.stopMethodTracing(); //结束 trace:
使用DDMS
DDMS(Dalvik Debug Monitor Service),是AndroidSDK里面自带的工具,开发环境中对Dalvik虚拟机调试监控的一种服务,它用于对Android的应用程序以及Framework层的代码进行性能分析。具体使用方式可自行查阅资料,这里就不做过多补充了。
从上面几个简单的方法我们可以知道启动的总体耗时,那么具体是哪个操作执行耗时过长呢? 我们需要借助工具根据实际情况去分析。Android Studio CPU Profiler 是Google 官方提供的检测工具,我们平常开发中可利用它来分析应用启动过程中cpu执行耗时详情信息。需要注意的是,由于是这种方式是侵入式的,实际耗时会有些许失真,收集到的时长会比真实时间要长一些,不过我们可以通过整体耗时比例及触发业务场景,识别性能瓶颈,去对症下药优化应用启动时间。
小结: 应用启动耗时的具体症结可以通过各种工具来监测,上述只是常用的一些工具,同时Android studio不同的版本 工具可能也会更新和完善,这里就不过多具体阐述工具的使用了。
从上述几个方面的分析,我们可以知道应用启动耗时的一些监测方式,通过监测我们可以查找具体症结所在,然后对症下药,接下来我们再讨论一下常见优化解决方案,提供思路及方案以供参考。
相信大家都看到过这种情况:应用启动时,有时会出现短暂黑屏或白屏的现象。如果启动比较慢的时候,白屏/黑屏过久甚至会长达几秒,这严重影响了用户的体验。那么为什么会出现黑屏或白屏?我们又应该怎么解决这个问题呢?具体缘由我们可以探索源码来仔细分析。
从上图我们可以看出,启动过程窗口的创建最终是交由PhoneWindowManager去管理的,那么我们下载source code 去看一下 PhoneWindowManager 对于主题设置的相关逻辑:
public class PhoneWindowManager implements WindowManagerPolicy {
//省略代码···
/** {@inheritDoc} */
@Override
public View addStartingWindow(IBinder appToken, String packageName, int theme,
CompatibilityInfo compatInfo, CharSequence nonLocalizedLabel, int labelRes,
int icon, int logo, int windowFlags, Configuration overrideConfig) {
if (!SHOW_STARTING_ANIMATIONS) {
return null;
}
if (packageName == null) {
return null;
}
WindowManager wm = null;
View view = null;
try {
Context context = mContext;
if (DEBUG_STARTING_WINDOW) Slog.d(TAG, "addStartingWindow " + packageName
+ ": nonLocalizedLabel=" + nonLocalizedLabel + " theme="
+ Integer.toHexString(theme));
if (theme != context.getThemeResId() || labelRes != 0) {
try {
context = context.createPackageContext(packageName, 0);
context.setTheme(theme);
} catch (PackageManager.NameNotFoundException e) {
// Ignore
}
}
if (overrideConfig != null && overrideConfig != EMPTY) {
if (DEBUG_STARTING_WINDOW) Slog.d(TAG, "addStartingWindow: creating context based"
+ " on overrideConfig" + overrideConfig + " for starting window");
final Context overrideContext = context.createConfigurationContext(overrideConfig);
overrideContext.setTheme(theme);
final TypedArray typedArray = overrideContext.obtainStyledAttributes(
com.android.internal.R.styleable.Window);
final int resId = typedArray.getResourceId(R.styleable.Window_windowBackground, 0);
if (resId != 0 && overrideContext.getDrawable(resId) != null) {
// We want to use the windowBackground for the override context if it is
// available, otherwise we use the default one to make sure a themed starting
// window is displayed for the app.
if (DEBUG_STARTING_WINDOW) Slog.d(TAG, "addStartingWindow: apply overrideConfig"
+ overrideConfig + " to starting window resId=" + resId);
context = overrideContext;
}
}
//省略代码···
}
}
分析探索源码过程中我们可以知道,App启动闪黑屏/白屏的原因在于PhoneWindowManager中的addStartingWindow 方法里的设置逻辑,从addStartingWindow方法中我们不难看出 系统进程在创建Application的过程中会产生一个BackgroudWindow,直到完成第一次绘制,系统进程才会用MainActivity的界面背景替换掉原来的占位BackgroudWindow。
从点击Lunach Icon那一刻起,到系统调用Activity.onCreate()之间的这个时间段内,WindowManager会先加载app主题样式中的windowBackground做为App的预览元素,然后再真正去加载Activity的layout布局。很显然,出现启动白屏或黑屏的情况(取决于主题是Dark还是Light),是因为我们的Application或Activity启动的这个过程太耗时,从而导致系统默认的BackgroundWindow没有及时被替换。
经过上述分析,那么问题就迎刃而解了。应用启动时黑屏或白屏过久的现象,无非是因为应用启动时WindowManager会去加载app主题样式中的windowBackground 而这个背景是根据当前应用的主题背景色决定的。那么我们有两种解决办法:
透明主题参考代码:
<style name="AppTheme" parent="Theme.AppCompat.Light.DarkActionBar">
- "android:windowFullscreen"
>true
- "android:windowIsTranslucent">true
style>
图片背景主题参考代码:
<style name="AppTheme" parent="Theme.AppCompat.Light.NoActionBar">
- "android:windowBackground"
>@drawable/launch
- "android:windowFullscreen">true
- "android:windowDrawsSystemBarBackgrounds">false
style>
可根据实际情况酌情使用,需要注意的是这种修改主题背景色的办法只能优化视觉效果,对于启动耗时没有什么本质性帮助。
注:启动窗口源码部分可以参考老罗的那篇博客,链接在本文末尾。
随着移动端的发展,诸如Push、LBS、Share、HotFix等功能可以说是应用必备功能,而这也催生了大量第三方SDK的横空出世,为广大开发者带来了福音,帮助开发者节省了很多开发时间和精力。但也由于这样,随着时间推移,越来越多的第三方SDK 被引入进我们的项目中,而这些第三方SDK 大多都在Application创建时被集中初始化,导致Application的启动耗时过久。应用初次安装启动时,严重的情况下,甚至可能出现启动时间长达十几秒甚至几十秒的现象,甚至出现启动ANR。
Google官方对它的定义 —— Avoid Heavy App Initialization,主要涉及到以下几点:
根据应用实际业务情况,可考虑将闪屏页/广告页等改成 Fragment,一般来说大约可减少启动耗时100ms左右。但需要注意业务改动成本及生命周期问题。
对于一些更新频率比较低的配置信息,或者资源等,我们可以采用缓存的方式避免每次启动都去下载,从而节省启动时间和CPU资源。
针对业务相关的代码逻辑,我们主要从下述几个方面去摸索优化之路:
小结: 相对而言,业务相关的代码可优化空间通常是比较大的,但复杂程度也比较高。需要我们有耐心地对应用针对性抽丝剥茧,整理业务。
保活可以降低应用冷启动概率,让APP变成温启动,这样可以大大减少Application 创建及初始化耗时。对于QQ、微信、淘宝等大厂应用来说,一般都可以寻求与手机厂商合作,通过应用白名单,或是定制优化应用启动时间。对于中小应用而言,则更多地是通过各种黑科技实现保活机制,不过这种方式也造成了Android生态圈的各种问题,并且Android 8.0以后Google也提高了限制,保活机制开始变得越来越难。在低版本系统可根据自身应用实际情况做一些保活手段。一些比较简单的常见手法是提高进程优先级,或利用守护进程;以及拦截系统返回键双击回退的处理逻辑等。
众所周知,我们的Android应用是运行在Java虚拟机之上的,而Java虚拟机进行垃圾回收的时候,要做一件很形象的事,叫做stop-the-world。也就是说,为了回收那些不再使用的对象,虚拟机必须要停止所有的线程来进行必要的回收工作。虽然这一点在ART得到了很大的改善,但GC是有代价的,它对App运行时的性能始终会有影响(诸如内存抖动等问题)。
优化建议:
小结: 总体而言,我们需要尽量避免频繁的GC,减少它触发的次数。以免造成主线程长时间卡顿。
对于Android 开发者来说, 想必我们对Dex 都不会太陌生,它承载了类以及APK里面的各种资源文件,APP在安装以及启动过程中都会读取dex文件。虽然Dex 里面包含的文件一般都比较小,但它们的读取频率非常高。因此,我们可以想办法去优化dex的排序以及分包、类加载等逻辑来提高系统的I/O效率,提高启动速度。
可以利用 ReDex 来优化我们的APK结构及体积,它是一款由Facebook 开源的工具,通过对字节码进行优化以减小Android Apk 的大小,同时可提高 App 启动速度。ReDex GitHub地址:https://github.com/facebook/redex
ReDex优化主要有以下几方面:
当然,接入ReDex有一定成本和风险,我们也可以根据实际参考facebook的思路如Interdex等,自研一套优化方案(没错,腾讯和微信团队都是这么干的)。此外,应用加固、热修复、插件化等方案对于启动速度也有比较大的影响。例如Tinker的热修复方案,大概会让启动速度增加6%-10%左右(粗略统计)。因此我们难免需要作出一些取舍,根据实际业务去做权衡利弊。
本文主要讲述了APP启动耗时的定义、检测以及分析、常规解决方案等内容,通过学习我们了解了启动优化以及耗时检测的常见方法和套路。实际上,有条件的话,我们也可以对上报统计系统增加一个启动耗时的功能,针对线上用户进行监测及收集并有针对性地去优化启动速度,同时也可以根据实际业务情况对低中高端机型做差异性优化。总的来说,我们做优化工作需要考虑到的方面比较多,有时需要从细节去突破。另外,做优化不能单单看KPI指标,需要从本质上出发,为用户真实的使用体验角度去考虑问题。
参考资料:
《Android窗口管理服务WindowManagerService显示Activity组件的启动窗口(Starting Window)的过程分析》 - 罗升阳
《 Optimizing Android bytecode with ReDex 》 - Facebook
《Redex初探与Interdex:Andorid冷启动优化》 - 【腾讯Bugly干货分享】