随着App的逻辑不断庞大,一不注意就会将耗时的操作放置在应用启动过程之中,导致应用启动速度越来越慢,用户体验也越来越差。优化启动速度是几乎所有大型App应用开发者需要考虑的问题。优化启动速度之前首先需要准确测量App启动时间,这样有利于我们更准确可量化地看出优化效果,也可以指导我们进行持续优化。
使用命令行方式统计多次启动某个Activity的平均用时可以在shell中执行如下指令:
adb shell am start -S -R 10 -W com.example.app/.MainActivity
其中-S
表示每次启动前先强行停止,-R
表示重复测试次数。每一次的输出如下所示信息。
Stopping: com.example.app
Starting: Intent { act=android.intent.action.MAIN cat=[android.intent.category.LAUNCHER] cmp=com.example.app/.MainActivity }
Status: ok
Activity: com.example.app/.MainActivity
ThisTime: 1059
TotalTime: 1059
WaitTime: 1073
Complete
其中TotalTime
代表当前Activity启动时间,将多次TotalTime
加起来求平均即可得到启动这个Activity的时间。
<intent-filter>
或者属性没有android:exported="true"
的Activity不能使用这种命令行的形式计算启动时间。
以上基于命令行的方式存在诸多问题,迫使我们思考怎样才能得到从用户角度上观察更准确的启动时间。在尝试其他方法之前,我们先定义一下怎样才是从用户角度上观察的启动时间。
要优化以及分析启动时间,需要先了解App的启动流程。以冷启动为例子,Application以及Activity的启动流程如下,参考文章[3][4][5][6]:
更为直观和简单的流程图参考Colt McAnlis在Android Performance Patterns Season 6中的表述。有兴趣的同学可以点击链接看看(Youtube链接)。
从流程图以及参考Colt McAnlis的Android Performance Patterns[6]得知,在冷启动的过程中,首先会通过AMS在System进程展示一个Starting Window(通常情况下是个白屏,可以通过设置Application的theme修改),接着AMS会通过Zygote创建应用程序的进程,并通过一系列的步骤后调用Application的attachBaseContext()
、onCreate()
然后最终调用Activity的onCreate()
以及进行View相关的初始化工作。在Activity展示出来后会替换掉之前的Starting Window,这样启动过程结束。
参考[1]发现在Activity中onWindowFocusChanged()
方法是最好的Activity对用户可见的标志,因此综合上一节的分析,我们可以考虑在Application的attachBaseContext()
方法中开始计算冷启动计时,然后在真正首页Activity的onWindowFocusChanged()
中停止冷启动计时,这样就可以初步得到应用的冷启动时间。
public void onWindowFocusChanged(boolean hasFocus)
Called when the current
android.view.Window
of the activity gains or loses focus. This is the best indicator of whether this activity is visible to the user.
为了方便统计,设置一个Util类专门做计时,添加的代码如下:
/**
* 计时统计工具类
*/
public class TimeUtils {
private static HashMap<String, Long> sCalTimeMap = new HashMap<>();
public static final String COLD_START = "cold_start";
public static final String HOT_START = "hot_start";
public static long sColdStartTime = 0;
/**
* 记录某个事件的开始时间
* @param key 事件名称
*/
public static void beginTimeCalculate(String key) {
long currentTime = System.currentTimeMillis();
sCalTimeMap.put(key, currentTime);
}
/**
* 获取某个事件的运行时间
*
* @param key 事件名称
* @return 返回某个事件的运行时间,调用这个方法之前没有调用 {@link #beginTimeCalculate(String)} 则返回-1
*/
public static long getTimeCalculate(String key) {
long currentTime = System.currentTimeMillis();
Long beginTime = sCalTimeMap.get(key);
if (beginTime == null) {
return -1;
} else {
sCalTimeMap.remove(key);
return currentTime - beginTime;
}
}
/**
* 清除某个时间运行时间计时
*
* @param key 事件名称
*/
public static void clearTimeCalculate(String key) {
sCalTimeMap.remove(key);
}
/**
* 清除启动时间计时
*/
public static void clearStartTimeCalculate() {
clearTimeCalculate(HOT_START);
clearTimeCalculate(COLD_START);
sColdStartTime = 0;
}
}
然后在Application的attachBaseContext()
方法中添加如下代码:
@Override
protected void attachBaseContext(Context base) {
super.attachBaseContext(base);
if (/**如果是主进程**/) {
TimeUtils.beginTimeCalculate(TimeUtils.COLD_START);
}
}
在第一个Activity的onCreate()
方法中添加如下代码:
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
calculateStartTime();
....
}
private void calculateStartTime() {
long coldStartTime = TimeUtils.getTimeCalculate(TimeUtils.COLD_START);
// 这里记录的TimeUtils.coldStartTime是指Application启动的时间,最终的冷启动时间等于Application启动时间+热启动时间
TimeUtils.sColdStartTime = coldStartTime > 0 ? coldStartTime : 0;
TimeUtils.beginTimeCalculate(DictTimeUtil.HOT_START);
}
在真正的首页Activity的 onWindowFocusChanged()
方法中添加如下代码:
@Override
public void onWindowFocusChanged(boolean hasFocus) {
if (hasFocus && /**没有经过广告或者引导页**/) {
long hotStartTime = TimeUtils.getTimeCalculate(TimeUtils.HOT_START);
if (TimeUtils.sColdStartTime > 0 && hotStartTime > 0) {
// 真正的冷启动时间 = Application启动时间 + 热启动时间
long coldStartTime = TimeUtils.sColdStartTime + hotStartTime;
// 过滤掉异常启动时间
if (coldStartTime < 50000) {
// 上传冷启动时间coldStartTime
}
} else if (hotStartTime > 0) {
// 过滤掉异常启动时间
if (hotStartTime < 30000) {
// 上传热启动时间hotStartTime
}
}
}
}
上面的分析给了我们初步的加log的起始和结束点,然而在实际的统计中会发现得到的数据有20%左右是不准确的,体现在计时数据非常大,有些甚至会显示冷启动时间超过一天。经过分析,在计算启动计时的时候需要注意一些问题。以下列举一下添加log时候需要注意的checklist。
应用在启动过程可能会有广告(我们的业务是有道词典),第一次启动会有引导页,需要根据业务情况标记在没有广告、没有引导页的时候才计算。这种情况要注意在非正常启动的时候忽略启动时间统计。
由于词典首页之前还有几个Activity,在没到首页Activity之前如果过早的返回,会出现冷启动时间过长的问题。这是因为词典返回的时候并没有杀掉进程,而时间统计信息是保存在内存中的,而等下次再进入的时候因为是热启动不会重新开始冷启动计时。这导致了这次热启动实际上打log的时候发现有上次冷启动的开始时间,算成了冷启动,而且因为启动时间是上一次的,所以这次冷启动log的时间比实际时间长。这种情况要注意在首页Activity之前的其他ActivityonPause()
方法中调用TimeUtils.clearStartTimeCalculate();
清除计时。
除了正常的启动流程,应用还有很多可能会导致Application的创建的入口,例如点击桌面小插件、系统账号同步、Deep Link跳转、直接进入设置了<action android:name="android.intent.action.PROCESS_TEXT" />
的Activity、push达到等。我们需要检查所有有可能引起Application创建,但是不是正常启动流程的地方,调用TimeUtils.clearStartTimeCalculate();
清除计时,避免引起冷启动时间计算过长错误的问题。
为了测试启动的过程中哪些方法比较耗时,我们可以使用Android Studio中集成的Android Monitor提供的Method Tracering或者Systrace。不过在实践中发现,有另外一个nimbledroid工具使用更加简便且能更明确指出耗时的地方。上传了应用之后会自动分析情景如下图所示。其中会自动检测出首页的Activity并且给出冷启动的启动情况。
点击进入Cold Startup的情景可以看到主要耗时的方法如下图。
至于为什么nimbledroid会知道那个是我们首页的Activity,官网上解析如下:
We use a heuristic to tell when an app finishes startup by detecting when (1) the main Activity has been displayed and (2) things like animated progress bars in the main Activity have stopped. Based on our experiments, this heuristic works in most cases.
点击进入某个方法,可以看到这个方法具体是由于调用了哪个子方法导致了耗时的问题。
通过nimbledroid这个工具,我们可以比较轻松地发现一些比较明显的问题,并可以指导我们进行启动优化。同时nimbledroid还支持Memory Leaks、网络监测以及结果分享等一些功能,更多的功能有待读者继续发现。
统计和分析启动时间有利于指导我们优化启动时间。以上介绍了有道词典在进行启动优化中的分析过程。通过详细了解Android应用启动的流程,进行准确的log记录,并且结合第三方工具,我们最终得到准确的启动时间统计数据以及启动优化的一些头绪。
【1】单刀土豆,2016.Android 开发之 App 启动时间统计
【2】Android Developer,Launch-Time Performance
【3】./multi_core_dump,2010.Android Application Launch
【4】./multi_core_dump,2010.Android Application Launch Part 2
【5】罗升阳,2012.Android系统源代码情景分析
【6】Colt McAnlis,2016.Android Performance Patterns Season 6
本文来自网易实践者社区,经作者申国骏授权发布。