众所周知,在Manifest的activity标签中有这么一个配置项android:launchMode,就是它决定着此activity的启动模式!
总共有四种模式,他们与Intent对象中的活动标志(FLAG_ACTIVITY_ *常量)结合使用,以确定Activity被唤起处理Intent时应该发生的情况。 他们是:
standard: 默认。系统始终会在目标任务中创建新的 Activity 实例,并向其传送 Intent。
singleTop:如果目标任务的顶部已存在 Activity 实例,则系统会通过调用该实例的 onNewIntent() 方法向其传送 Intent,而非创建新的 Activity 实例。
singleTask:系统会在新任务的根位置创建 Activity 并向其传送 Intent。不过,如果已存在 Activity 实例,则系统会调用该实例的 onNewIntent() 方法(而非创建新的 Activity 实例),向其传送 Intent。
singleInstance:与singleTask相同,只是系统不会将任何其他 Activity 启动到包含实例的任务中。该 Activity 始终是其任务中的唯一 Activity。
小知识点梳理
以standard和singleTop模式启动的Activity可以多次实例化,并且可以位于Activity栈中的任何位置。
相比之下,以singleTask和singleInstance模式启动的Activity 只能启动任务。它们始终位于 Activity 栈的根位置。此外,设备一次只能保留一个 Activity 实例,即一次只允许一个此类任务。
singleTask和singleInstance模式同样只有一处不同:singleTask模式的Activity 允许其他 Activity 成为其任务的一部分。该 Activity 始终位于其任务的根位置,但其他 Activity(必然是standard和singleTop模式的Activity)可以启动到该任务中。另一方面,singleInstance模式的Activity 不允许其他 Activity 成为其任务的一部分。它是任务中唯一的 Activity。如果它启动另一个 Activity,则系统会将该 Activity 分配给其他任务,就如同 Intent 中包含 FLAG_ACTIVITY_NEW_TASK 一样。
实用命令:
可以查看栈信息
adb shell dumpsys activity activities
android:taskAffinity
taskAffinity这个配置项和启动模式有着千丝万缕的联系。可以翻译成任务的相似性。从概念上讲,具有同一相似性的 Activity 归属同一任务(从用户的角度来看,则是归属同一“应用”)。任务的相似性由其根 Activity 的相似性确定。
相似性确定两点内容 — Activity 更改父项后的任务(请参阅 allowTaskReparenting 属性),以及通过 FLAG_ACTIVITY_NEW_TASK 标记启动 Activity 时,用于容纳该 Activity 的任务。
默认情况下,应用中的所有 Activity 都具有同一相似性。您可以设置该属性,以不同方式将其分组,甚至可以在同一任务内放置不同应用中定义的 Activity。如要指定 Activity 与任何任务均无相似性,请将其设置为空字符串。
如果未设置该属性,则 Activity 会继承为应用设置的相似性(请参阅 <application> 元素的 taskAffinity 属性)。应用默认相似性的名称为 <manifest> 元素所设置的软件包名称。
android:allowTaskReparenting
当下一次将启动 Activity 的任务转至前台时,Activity 是否能从该任务转移至与其有相似性的任务 —“true”表示可以转移,“false”表示仍须留在启动它的任务处。
如果未设置该属性,则对 Activity 应用由 <application> 元素的相应 allowTaskReparenting 属性所设置的值。默认值为“false”。
正常情况下,Activity 启动时会与启动它的任务关联,并在其整个生命周期中一直留在该任务处。当不再显示现有任务时,您可以使用该属性强制 Activity 将其父项更改为与其有相似性的任务。该属性通常用于将应用的 Activity 转移至与该应用关联的主任务。
例如,如果电子邮件消息包含网页链接,则点击该链接会调出可显示该网页的 Activity。该 Activity 由浏览器应用定义,但作为电子邮件任务的一部分启动。如果将该 Activity 的父项更改为浏览器任务,则它会在浏览器下一次转至前台时显示,在电子邮件任务再次转至前台时消失。
Activity 的相似性由 taskAffinity 属性定义。通过读取任务根 Activity 的相似性即可确定任务的相似性。因此,按照定义,根 Activity 始终位于具有同一相似性的任务中。由于具有“singleTask”或“singleInstance”启动模式的 Activity 只能位于任务的根,因此更改父项仅限于“standard”和“singleTop”模式。(另请参阅 launchMode 属性。)
启动标志位
提到启动模式,Intent中的启动标志位就是不得不看的一个内容了。
- FLAG_ACTIVITY_NEW_TASK
这个标志位可以算是大名鼎鼎了。相对也比较熟悉。当需要在一个新的任务中启动Activity时,加上次标志位。
Launcher在启动应用的时候就会统一加上这个标志位。
如果使用Service启动Activity的话,也需要加上这个。
当使用此标志位时,如果有一个任务已经在运行你想要启动的Activity,此时,不会有一个新的Activity被启动,取而代之的是,当前这个任务将简单地以其上次进入的状态显示在屏幕的前面。有关禁用此行为的标志,请参见FLAG_ACTIVITY_MULTIPLE_TASK。
当调用者从正在启动的Activity中请求结果时,不能使用此标志。 - FLAG_ACTIVITY_CLEAR_TOP
如果设置了这个标志位,那么在启动一个Activity时,如果这个Activity已经在当前任务中了,那么所有在这个Activity以上的Activity都会被关闭,这个Intent会以new Intent的形式被分发给旧的那个Activity,当然,在这个旧Activity头上的兄弟被关闭之后,他现在已经在栈顶了。
举个例子:
有任务栈中已经存在4个Activity A、B、C、D,其中A在栈底,D在栈顶,也就是说显示的是Activity D。此时Activity D用FLAG_ACTIVITY_CLEAR_TOP的标志位启动Activity B,那么C和D会被finish掉,B会收到新的Intent,最后栈里之身下A、B。
对于上面例子中的Activity B,在他被启动时,要么通过onNewIntent方法,要么通过finish之后重启。如果他的启动模式被声明为"multiple" (the default),同时Intent中没有设置FLAG_ACTIVITY_SINGLE_TOP,那么他将被重启。对于其他的启动模式,或者设置了FLAG_ACTIVITY_SINGLE_TOP的情况,他的onNewIntent方法会被调用。
此启动模式也可以与FLAG_ACTIVITY_NEW_TASK结合使用,以达到良好效果:如果用来启动任务中的根Activity,它将把该任务的任何当前正在运行的实例置于前台,然后将其清除为根状态。例如,当从通知栏启动Activity时,此功能特别有用。 - FLAG_ACTIVITY_SINGLE_TOP
和singleTop启动模式一致