你的应用,启动够快吗?
App 启动时,我们通常会执行很多既定的任务,比如各种 SDK 的初始化、各种数据的初始化等。这必将拖慢 App 的启动速度。
启动速度,一直在我们的 App 优化清单上,而在 11 月举办的 Android 绿色联盟开发者大会上,推出了应用体验标准。对应用的兼容性、稳定性、性能、功能和安全,都做了详细的定义。
其中启动速度做了更细致的要求,冷启动 < 1s,热启动 < 0.5s。这当然是最理想的情况,具体还有外部因素的影响,例如手机配置。
启动时初始化的代码,势必会拖慢 App 的启动速度,但是不等待这些任务初始化完成呢,可能又会造成功能的缺失,而有时候不同任务还存在时序依赖的关系,要求我们先执行 A 任务,再执行 B 任务。
每次都特殊处理会增加复杂度,我们需要一套通用的解决方案。
这里给大家推荐一个 Alibaba 在 Github 上开源的一个 Alpha 的库。它是一个基于 PERT 图构建的 Android 异步启动框架。
Alibaba Alpha
PERT 图又称为“计划评审技术”,它采用网络图来描述一个项目的任务网络。不仅可以表达任务与任务之间的关系,不同的任务之间还可以有各种子任务。
Alpha 就是基于 PERT 图设计思想,它简单、高效、功能完整。利用 Alpha 在应用启动的时候,让这些任务并发处理,从而达到提高启动速度的目的。并且可以保证他们执行顺序的正确性。
Alpha 在使用时,只需要定义好自己的 Task,并描述它依赖的 Task,再将他们添加到 Project 中。框架会自动并发有序地执行这些 Task,最后将执行结果抛出来。
为了支持 Android 应用的多进程,Alpha 支持为不同进程配置不同的启动模式。
Alpha 的接入非常简单,它支持 Java 代码和配置文件两种方式来构建一个启动流程。
接入 Alpha
Gradle 依赖
compile 'com.alibaba.android:alpha:1.0.0.1@jar'
使用 Gradle 可能会导致失败,这里可以考虑通过源码引入的方式。
使用指南
1. 实现自己的 Task 类
在 Alpha 中,任务都是一个个 Task。定义一个 Task,并在 run() 方法中实现该 Task 需要做的事情。
public class SampleTask extends Task{
public SampleTask() {
super("SampleTask");
}
@Override
public void run() {
//do something, print a msg for example.
Log.d(TAG, "run SampleTask");
}
}
Task 默认是通过异步的方式在子线程中执行,如果这个 Task 需要在主线程中执行,可以在构造函数中指定。
/**
* 构造{@code Task}对象。
*
* @param name {@code Task}名字
* @param isInUiThread 是否在UI线程执行,true表示在UI线程执行,false表示在非UI线程执行,默认在非UI线程执行。
* <strong>注意:如果在UI线程执行,则不能再使用{@link AlphaManager#waitUntilFinish()},否则会造成死锁。</strong>
*/
public Task(String name, boolean isInUiThread) {
mName = name;
mIsInUiThread = isInUiThread;
}
2. 将 Task 组合成一个完整的 Project
可以用 Task.ProjectBuilder 依据各 Task 之间的依赖关系,将这些 Task 构建成一个完整的 Project。
private Task createCommonTaskGroup() {
Task a = new TaskA();
Task b = new TaskB();
Task c = new TaskC();
Task d = new TaskD();
Task e = new TaskE();
Project.Builder builder = new Project.Builder();
builder.add(a);
builder.add(b).after(a);
builder.add(c).after(a);
builder.add(d).after(b, c);
builder.add(e).after(a);
Project group = builder.create();
return group;
}
ProjectBuilder 生成的 Project 本身又可以作为一个 Task 嵌入到另一个 Project 中。
private Task createCommonTaskGroup() {
Task a = new TaskA();
Task b = new TaskB();
Task c = new TaskC();
Task d = new TaskD();
Task e = new TaskE();
Project.Builder builder = new Project.Builder();
builder.add(a);
builder.add(b).after(a);
builder.add(c).after(a);
builder.add(d).after(b, c);
builder.add(e).after(a);
Project group = builder.create();
return group;
}
private void createProject() {
Task group = createCommonTaskGroup();
Task f = new TaskF();
Project.Builder builder = new Project.Builder();
builder.add(group);
builder.add(f);
Project project = builder.create();
}
3. 监听 Task 执行结束
可以通过 addOnTaskFinishListener() 监听 Task 的执行结束。
/**
* <p>增加{@code Task}执行结束的监听,当该{@code Task}执行结束时,会回调
* {@link Task.OnTaskFinishListener#onTaskFinish(String)}。</p>
* <strong>注意:</strong>回调函数在{@code Task}所在线程中回调,注意线程安全。
*
* @param listener 监听{@code Task}执行结束的{@code listener}
*/
public void addOnTaskFinishListener(OnTaskFinishListener listener) {
if (!mTaskFinishListeners.contains(listener)) {
mTaskFinishListeners.add(listener);
}
}
4. 为构建完成的 Project 配置对应的进程
通过 addProject(),将 Project 配置到对应的进程中。
AlphaManager.getInstance(mContext)
.addProject(project);
5. 执行启动流程
最后只需要调用一句 start() 方法,就可以执行这个完整的流程了。
AlphaManager.getInstance(mContext)
.start();
Alpha 还提供了配置文件的方式来配置 Task 的关系,但是我不准备再详细介绍了,反正我不会这么用,有兴趣还是去查询 README 文件吧。
小结
利用 Alpha 我们可以有效的管理启动时初始化的一些任务,从而达到优化启动速度的目的。
#p#分页标题#e#Alpha 是 Alibaba 开源的,现在我还不确定用在了那些阿里系的商业项目上,不过从源码的角度来看,没有大坑,而且各自文档也很齐全,如果有这方面的需求,可以尝试使用。
【本文为51CTO专栏作者“张旸”的原创稿件,转载请通过微信公众号联系作者获取授权】
戳这里,看该作者更多好文
【编辑推荐】
为 TV 开发的 App,你说要运行在手机上?
Flutter:一小时从零构建一个简单的 App,以及你如何做到这一点!
漫画:App 防止 Fiddler 抓包小技巧!
Spring Boot 2.0 新特性(二):新增事件ApplicationStartedEvent
从大团队并肩作战到小团队带头冲锋,苏宁App插件化应用实践