深入排查 FLAG 导致的 Activity 无法正常启动

dc0d7012e613989d99514c0a40a286ca.png

bc72cc6990918e307d2e3efc57221ecc.gif 

本文字数: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

复现流程

将所有不影响异常的业务代码删除之后得到的最精简复现流程如下:

  1. Service 附加 FLAG_ACTIVITY_NEW_TASK 启动 A_Activity:

    startActivity(Intent(this, AActivity::class.java).apply { addFlags(Intent.FLAG_ACTIVITY_NEW_TASK) })
  2. A_Activity 启动 B_Activity:

    startActivity(Intent(this, BActivity::class.java))
  3. 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 均未回调 onCreateonResume 或者 onNewIntent 等生命周期函数,也没有任何 Crash 或者 ANR 的日志发生,点击 B_Activity 可以正常操作。

原因推测

首先排除 Service 的 Context 的原因,因为我们可以构造如下流程,实验会发现 A_Activity 依然无法启动:

  1. MainActivity 中先 finish 后附加 FLAG_ACTIVITY_NEW_TASK 启动 A_Activity:

    finish()
    startActivity(Intent(this, AActivity::class.java).apply { addFlags(Intent.FLAG_ACTIVITY_NEW_TASK) })
  2. A_Activity 启动 B_Activity:

    startActivity(Intent(this, BActivity::class.java))
  3. B_Activity 附加 FLAG_ACTIVITY_NEW_TASK 启动 A_Activity:

    startActivity(Intent(this, AActivity::class.java).apply { addFlags(Intent.FLAG_ACTIVITY_NEW_TASK) })

那么从使用排除法得到 FLAG

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值