Android中的LaunchMode

写这篇文章主要参考https://www.bilibili.com/video/BV1CA41177Se
和《Android开发艺术探索》这篇文章是我学习LaunchMode的学习笔记

众所周知,Android是使用回退栈来管理Activity的,而根据Android中的LaunchModeActivity入栈的方式不同

Android中的LaunchMode分为四种:Standard,SingleTask,SingleInstance,SingleTop

首先介绍下什么是Task?每个回退栈都对应着一个Task,当我们查看最近任务列表时查看的其实就是一个个的Task,当我们在手机桌面点击一个app进入这个app的laucherActivity时安卓系统会创建一个Task并且将这个launcherActivity放入这个task中,当我们在这个laucherActivity启动一个启动模式是Standard的Activity时,这个Activity会被放在laucherActivity的上方,当我们按back键时这个新打开的activity就会从这个task中退出从而laucherActivity又位于这个task的最上方,再按back就会回到桌面,此时我们查看最近任务列表还是能看到这个laucheractivity的页面,但是只是task还存在,activity实际上已经销毁了,我们从最近任务列表再次进入实际上跟在桌面点击app图标进入activity是一样的,所以在最近任务里看见的task 未必是活着的activity。默认情况下当你在a app 里打开b app 的activity 的时候,这个b app 的activity 会直接被放进a 的task 里,因为Android默认在不同Task中打开同一个Activity时,Activity会被创建多个实例,分别放进每一个Task。

再来说说什么是taskAffinity?默认情况下,所有Activity都有一个taskAffinity属性且值是应用的包名,我们也可以为每个Activity单独指定taskAffinity属性,这会导致Activity进入Task的方式也不同,这个在后面慢慢分析。

最后挨个分析这四种启动模式

Standard

这是系统默认的启动模式,每次启动一个Activity都会创建一个新的实例,不管这个实例是否已经存在,在这种模式下,谁启动了这个Activity,那么这个Activity就运行在启动它的那个Activity里,这对应了前面所说的Android默认在不同Task中打开同一个Activity时,Activity会被创建多个实例,分别放进每一个Task.

SingleTask

这种模式保证了一个Activity在全局中只有一个实例,

当我们在在a app 里打开b app 的activity 的时候,如果这个b app 的activity的LaunchMode是SingleTask的话,那么系统此时就会创建一个taskAffinity为B的taskAffinity的Task并将这个Task压在a appTask的上面,所以此时我们相当于打开了Bapp,会出现Bapp的入场动画,如果是在b app 的activity的LaunchMode是默认情况下的话,b app 的activity是直接进入了a app的task里。

但是这种方式有一个问题,b app的task压在a app的task上时,如果我们此时查看最近任务列表或者按Home回到桌面,压在a app task上的b app 的task就不会再压在a app上,而是俩者处于一种平行关系了,此时我们再进入a app不会再显示我们看到我们打开的bapp的activity,而是a app的activity

怎么解决这个问题呢?

来说说Taskreparenting属性

在默认情况下,Activity归属于一个Task。不会在多个Task 之间跳来跳去。但你可以通过设置来改变这个逻辑。这就是Taskreparenting属性,我们给 bapp的activity设置这个属性并把launchmode设置为默认,当我们在在a app 里打开b app 的activity 的时候,b app的这个activity会被放在a app task的最上面,当我们打开b app时,这个原来放在a app task中的b的activity就会移动到此时b app中的task中。

SingleInstance

SingleInstance是更加严格的SingleTask,SingleTask保证了一个Activity在全局中只有一个实例,而SingleInstance在此基础上还要求一个栈中只能有一个这个Activity。具有此种模式的Activity只能单独的位于任务栈中,除非这个任务栈被销毁否则不会在创建新的Activity。

当这个Activity再次被启动时会调用它的onNewIntent方法。

设想这样一种情况,当我们在a app中打开b app的某个LaunchMode是SingleInstance的activity时,然后在这个新打开的activity中打开b app的另一个activity,那么由于SingleInstance的限制,此时会新建一个b app的task并且将这个新建的activity放入然后将bapp的task压在最上面,也就是此时有三个task压在一起。

再来看看这种情况,当我们在a app中打开b app的某个LaunchMode是SingleInstance的activity时,然后再按home键,再打开b app,再查看最近任务列表,会发现找不到我们再a app 打开b app的那个LaunchMode是SingleInstance的activity的task,这是为什么呢?

这是因为他们的taskAffinity冲突了

请看下图:

android 11 多个 launcher android launchmode_Standard


android 11 多个 launcher android launchmode_android_02

SingleTop

在这种模式下,如果新activity已经位于任务栈的栈顶,那么此Activity不会被重建,同时它的onNewIntent会被回调。

Intent Flag

在代码中通过StartActivity来启动下一个Activity的时候,可以通过设置Intent的Flag来修改默认的Task管理标志,如下:

Intent intent = new Intent(MainActivity.this, DetailActivity.class);
 intent.setFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP);


Flag的值有以下三个:
1)FLAG_ACTIVITY_NEW_TASK
这个值跟LaunchMode中的SingleTask的效果是一样的。
2)FLAG_ACTIVITY_SINGLE_TOP
这个值跟LaunchMode中的SingleTop的效果是一样的。
3)Flag_ACTIVITY_CLEAR_TOP
当Intent设置了这样一个标志,那么Android就会去寻找一个存在的Activity,如果找到了,其上面存在有其他Activity的话,就会将其上面的所有Activity都清除掉。

怎么选择?

android 11 多个 launcher android launchmode_android_03