本文字数:13665字
预计阅读时间:35分钟
深入源码排查 FLAG_ACTIVITY_NEW_TASK 导致的 Activity 无法正常启动
引言
众所周知 Android 中 Activity 有四种启动模式,决定了是否创建新的 Activity 实例或复用当前实例。但是实际上 Activity 的启动十分复杂,不单单是 android:launchMode
就能决定的,多个 Intent Flag 都能改变启动的行为模式,除此之外还会根据 android:taskAffinity
的值来判断对应任务栈是否存在。
基本 LaunchMode 行为模式
首先复习一下最基本的 LaunchMode 行为模式:
standard:默认值,每次启动都会创建一个新的 Activity 实例。
singleTop:如果待启动 Activity 处于栈顶则重用实例回调其
onNewIntent()
方法,否则新建实例。singleTask:如果已有任务栈中存在该 Activity 实例,则将实例之上的其他 Activity 清空并回调实例的
onNewIntent()
方法,否则新建栈创建 Activity 实例。singleInstance:在单独一个栈中创建,该栈不会再创建其他 Activity,当系统中存在实例时回调实例的
onNewIntent()
方法。
这是我们最熟知的启动模式,但是需要注意的是:上述行为模式仅限于未添加任何 Flag,并且没有改变 taskAffinity 的前提下。
Service 启动 Activity
如前所述,如果单纯的 Activity 启动 Activiy 且不添加任何 Flag 是没问题的,但是如果我们使用 Service 启动 Activity 就会遇到一个异常,在 Framework(本文 Framework 源码均来自 Android 11 版本)的源码中会抛出异常:
//android.app.ContextImpl
@Override
public void startActivity(Intent intent, Bundle options) {
……
if ((intent.getFlags() & Intent.FLAG_ACTIVITY_NEW_TASK) == 0
&& (targetSdkVersion < Build.VERSION_CODES.N || targetSdkVersion >= Build.VERSION_CODES.P)
&& (options == null || ActivityOptions.fromBundle(options).getLaunchTaskId() == -1)) {
throw new AndroidRuntimeException(
"Calling startActivity() from outside of an Activity "
+ " context requires the FLAG_ACTIVITY_NEW_TASK flag."
+ " Is this really what you want?");
}
……
}
所以在 Service 中启动 Activity 必须添加 FLAG_ACTIVITY_NEW_TASK
,原因也很简单,每个 Activity 启动都需要一个任务栈,非 Activity 的 context 存在后台启动的可能,而此时前台是其他 App 的任务栈,甚至我们的 App 根本没有创建过任务栈,为了防止这些无法预料的情况出现,被强制要求添加这个 Flag。
standard
是四种启动模式中最简单也最常见的,其他三种各有各的流程,唯独 standard
直接简单粗暴创建一个新的实例放到栈顶。那么问题来了,是否所有的 standard
都能创建一个新的实例呢?
无法启动的 standard
Activity
复现流程
将所有不影响异常的业务代码删除之后得到的最精简复现流程如下:
Service 附加
FLAG_ACTIVITY_NEW_TASK
启动 A_Activity:startActivity(Intent(this, AActivity::class.java).apply { addFlags(Intent.FLAG_ACTIVITY_NEW_TASK) })
A_Activity 启动 B_Activity:
startActivity(Intent(this, BActivity::class.java))
Service 附加
FLAG_ACTIVITY_NEW_TASK
启动 A_Activity:startActivity(Intent(this, AActivity::class.java).apply { addFlags(Intent.FLAG_ACTIVITY_NEW_TASK) })
会发现 A_Activity 无法被启动,A_Activity 和 B_Activity 均未回调 onCreate
、onResume
或者 onNewIntent
等生命周期函数,也没有任何 Crash 或者 ANR 的日志发生,点击 B_Activity 可以正常操作。
原因推测
首先排除 Service 的 Context 的原因,因为我们可以构造如下流程,实验会发现 A_Activity 依然无法启动:
MainActivity 中先
finish
后附加FLAG_ACTIVITY_NEW_TASK
启动 A_Activity:finish() startActivity(Intent(this, AActivity::class.java).apply { addFlags(Intent.FLAG_ACTIVITY_NEW_TASK) })
A_Activity 启动 B_Activity:
startActivity(Intent(this, BActivity::class.java))
B_Activity 附加
FLAG_ACTIVITY_NEW_TASK
启动 A_Activity:startActivity(Intent(this, AActivity::class.java).apply { addFlags(Intent.FLAG_ACTIVITY_NEW_TASK) })
那么从使用排除法得到 FLAG