Activity启动模式singleTask的理解

1. Activity启动模式singleTask的理解

2. Task与Activity栈

Task是一些Activity的集合,以Activity栈的形式存放。因此,Task是概念上的,Activity栈是实体上的。

可以说,新启动了一个Task就是新建了一个Activity栈,来存放这个Task内的Activity。

3. singleTask设置方式

  • manifest文件
    在activity节点里添加属性:
<activity
    android:name="com.example.test.SecondActivity"
    android:launchMode="singleTask"
/>

  • intent设置FLAG
Intent intent = new Intent(this, SecondActivity.class);
        intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
        startActivity(intent);

4. singleTask的意义(作用)

把一个activity启动模式设置为为singleTask,只是意味着framework在启动该activity时把它标识为可在一个新任务中启动,至于是否在一个新任务中启动,那可不一定了!是需要条件的,什么条件呢?请看下文分解~

4.1. 举例说明

4.2. 例子1

假设下面的跳转关系:

MainActivity -> SecondeActivity -> ThirdActivity

其中,SecondeActivity被设置了singleTask的启动模式,MainActivity和ThirdActivity都不做任何操作,保持默认,即standard模式

那么跳到ThirdActivity后,当前的Activity栈情况从顶到底部是:

ThirdActivity、SecondeActivity、MainActivity

也就是说依然按照正常压栈的顺序,和standard模式一样。并不是按照谷歌官方解释的,容易误解为新建了一个Task,并用新的栈存放SecondeActivity。这是误解。

那么,真的和standard模式一样吗?

还是有不同的,不然弄这个模式干嘛。

不同之处,用下面的例子说明:

4.3. 例子2

假设下面的跳转关系:

MainActivity -> SecondeActivity -> ThirdActivity -> ForthActivity ->SecondeActivity

其中,SecondeActivity被设置了singleTask的启动模式,其余Activity都不做任何操作,保持默认,即standard模式

那么跳到SecondeActivity后,当前的Activity栈情况从顶到底部是:

SecondeActivity、MainActivity

没错,SecondeActivity以上的 ThirdActivity、ForthActivity都被销毁了。

也就是说,重用了栈里面已有的 SecondeActivity ,并且把它上面的 activity 都清除了,从而使得它暴露在最上面。

顺便说一下,最后一个SecondeActivity的启动没有调用onCreate、onStart、onResume这些生命周期方法,而是调用了onNewIntent方法。

4.4. 小结

被标记为singleTask的Activity,在启动的时候如果当前没有实例,就和standard模式一样,直接压栈到当前activity上;如果已经有实例了,就把该目标Activity实例上面的activity都清除,从而将目标Activity暴露到最上面,调用了其onNewIntent方法

5. taskAffinity属性

上面的两个例子都在我们认知范围内,还比较熟悉。但是这个taskAffinity想必就不太熟悉了。

taskAffinity是标记当前activity附属到哪个task上的。

5.1. 配置方式

  • manifest文件里
    在activty节点里配置该属性即可。taskAffinity取值是一个字符串,用来唯一标识一个Task的。

默认情况下,也就是不配置,taskAffinity取的是应用的包名。

 <activity android:name="com.example.test.SecondActivity"
             android:launchMode="singleTask"
             android:taskAffinity="com.example.test.second">
        </activity>

5.2. 意义(作用)

配合singleTask使用,意味着被这两个属性标记的activity可以被指定到一个新的task里。

5.3. 举例

假设下面的跳转关系:

MainActivity -> SecondeActivity -> ThirdActivity

其中,SecondeActivity被设置了singleTask的启动模式,并且配置了taskAffinity属性,MainActivity和ThirdActivity都不做任何操作,保持默认,即standard模式

那么跳到ThirdActivity后,当前的Activity栈情况从顶到底部是:

Task1: ThirdActivity、MainActivity

Task2: SecondeActivity

可以看到,SecondeActivity被创建到了新的栈里。

于是,可以得出之前疑问的答案,什么条件下设置为singleTask的activity会被创建到新的task呢?

答案是,同时被指定了另一个taskAffinity属性值的时候。

5.4. 回退顺序

现在处于onResume的activity是ThirdActivity,如果这时候按返回键呢?activity的显示顺序会是什么样呢?

答案正如上面Task1、Task2栈的排列顺序,ThirdActivity弹出后,MainActivity显示;再按返回键,MainActivity弹出后,SecondeActivity显示。

可以看出,新建的Task处于下方。

5.5. 总结

  1. 把启动模式设置为singleTask,framework在启动该activity时只会把它标示为可在一个新任务中启动,至于是否在一个新任务中启动,还要受其他条件的限制,这个条件就是taskAffinity属性配置为不同于包名的其他字符串。如果没有配置,默认就是当前包名,在当前Task内操作。
  2. 在启动一个singleTask的Activity实例时,如果系统中已经存在这样一个实例,就会将这个实例调度到任务栈的栈顶,并清除它当前所在任务中位于它上面的所有的activity。
    如果不存在,就看taskAffinity属性对应的Task是否存在,如果存在,就把Activity压入该Task内栈里;否则,就新建一个Task,再把Activity压入该Task内栈里。
  3. singeTask模式保证了在一个Task内只有一个activity实例

参考文章:https://blog.csdn.net/zhangjg_blog/article/details/10923643

猜你喜欢

转载自blog.csdn.net/fxjzzyo/article/details/109293098