从今天开始,我将带领大家一起来解读Android 9.0的源码。在最近这几年,笔者的工作都是和framework的修改相关,对于这一块比较熟悉。因此,我力求能用最通俗易懂的语言,让刚开始接触源码的同学也能有所收获。为了描述方便,文中我们统一用AMS表示ActivityManagerService。废话不多说,我们直接进入正题。
在入门Android开发时,我们就已经知道如何去启动一个新的Activity了。我们以显式Intent为例,一般代码如下:
Intent intent = new Intent(this,ActivityXXX.class);
startActivity(intent);
或者
Intent intent = new Intent(this,ActivityXXX.class);
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
context.startActivity(intent);
第一种方式,一般我们是在Activity中这样使用;对于第二种方式,我们主要是在一些非界面的情况下使用,如BroadcastReceiver、Service等。现在我就分别来介绍这两种方式的启动流程是怎样的。
一、startActivity
(1)使用Activity启动
我们首先来看Activity.java中,关于startActivity是怎么定义的。
public class Activity extends ContextThemeWrapper implements ...{
...
@Override
public void startActivity(Intent intent) {
this.startActivity(intent, null);
}
@Override
public void startActivity(Intent intent, Bundle options) {
if (options != null) {
startActivityForResult(intent, -1, options);
} else {
// Note we want to go through this call for compatibility with
// applications that may have overridden the method.
//默认不加 options时,会走这个分支
startActivityForResult(intent, -1);
}
}
public void startActivityForResult(Intent intent, int requestCode){
startActivityForResult(intent, requestCode, null);
}
public void startActivityForResult(Intent intent, int requestCode, Bundle options) {
...
//启动目标Activity
Instrumentation.ActivityResult ar = mInstrumentation.execStartActivity(this,
mMainThread.getApplicationThread(), mToken, this,
intent, requestCode, options);
//发送执行结果
if (ar != null) {
mMainThread.sendActivityResult(mToken, mEmbeddedID, requestCode,
ar.getResultCode(), ar.getResultData());
}
...
//取消当前Activity的输入事件,并开始过渡退出
cancelInputsAndStartExitTransition(options);
...
}
...
}
从上面代码我们可以总结出,在startActivityForResult方法中,主要做了3件事情:
1、调用 Instrumentation.execStartActivity方法来启动Activity,并返回执行结果;
2、如果执行结果不为空,则通过ActivityThread. sendActivityResult方法,将执行结果返回到 Activity(或Fragment) 中;
这里的mMainThread是ActivityThread对象。在ActivityThread中,则会通过ClientTransaction、ActivityResultItem等类的中转,调用Activity的dispatchActivityResult方法, 并最终调用我们所熟悉的onActivityResult方法。关于ClientTransaction等,我们本文后续会再提到并详解;
3、取消当前Activity的输入事件,包括TouchEvent、KeyEvent 等,并开始退出。
(2)使用Context启动
我们都知道,在Android中,Context是一个抽象类,而其具体的实现类是ContextImpl。而至于Context其他相关类,如ContextWrapper,都是对ContextImpl的一种装饰(功能的增强),这是设计模式中典型的装饰模式。如果我们不了解装饰模式也没关系,目前知道Context的实现类是ContextImpl就行了。
我们来看ContextImpl中关于startActivity是如何定义的
public class ContextImpl extends Context {
...
@Override
public void startActivity(Intent intent) {
//如果是系统进程调用此方法,则打印 Log警告应使用startActivityAsUser
warnIfCallingFromSystemProcess();
//启动Activity
startActivity(intent, null);
}