一篇文章搞懂 Activity 启动模式

7,696 阅读3分钟

本文大部分翻译自 《Android “launchMode” (Visualized)》, 并结合自己的理解在 SingleInstance 中添加了补充内容

启动模式 是很重要的一块知识点,每位安卓开发都应该了解它是如何运作的。大部分的开发者(包括我自己)认为我们已经知道了如何来使用 启动模式。但是其实我们错了。不仅仅是知道将 Activity 设置为 SingleTask 来防止重复实例化就足够了,虽然大部分情况下这已经解决了我们的业务需求。但是我们真的了解当我们将一个 Activity 设置为 SingleTask 时,它是如何影响 Activity 任务栈内其他 Activites 的吗?

本文将图文的模式来讲解 4 中启动模式的作用及区别。首先需要来讲解以下图片的作用来帮助你更容易理解下面的内容

绿色代表在 Activity 任务栈中创建了一个新的 Activity 实例
黄色则代表复用了之前创建的 Activity 实例,但是该 Activity 的 onNewIntent() 方法将会被调用

Standard 启动模式讲解

如果在 AndroidManifest.xml 中将 Activity B 的启动模式设置为 Standard,不管任务栈内是否已经存在 Activity B 的实例,当启动 Activity B 时,都会创建一个崭新的 Activity B 位于任务栈顶

SingleTask 模式讲解

如果在 AndroidManifest.xml 中将 Activity C 的启动模式设置为 SingleTask,如果此时任务栈内已经存在 Activity C 的实例且未位于栈顶,当启动 Activity C 时,会将 Activity C 上方的实例全部出栈让其位于任务栈顶并 Activity C 中的 onNewIntent() 方法会被调用

如果在 AndroidManifest.xml 中将 Activity C 的启动模式设置为 SingleTask,并且此时任务栈内并不存在 Activity C 的实例,当启动 Activity C 时,会创建一个崭新的 Activity C 实例在栈顶

SingleTop 启动模式讲解

如果在 AndroidManifest.xml 中将 Activity D 的启动模式设置为 SingleTop 并且任务栈内存在 Activity D 实例且位于栈顶时,当启动 Activity D 时,会复用之前创建的 Activity D 的实例,并且 onNewIntent() 方法被调用

如果在 AndroidManifest.xml 中将 Activity D 的启动模式设置为 SingleTop 并且任务栈内并不存在 Activity D 的实例时,当启动 Activity D 时,会创建一个崭新的 Activity D 实例在栈顶

如果在 AndroidManifest.xml 中将 Activity D 的启动模式设置为 SingleTop 并且任务栈内存在 Activity D 的实例但其实例未在栈顶时,当启动 Activity D 时,会再次创建一个崭新的 Activity D 实例在栈顶

SingleInstance 模式讲解

如果在 AndroidManifest.xml 中将 Activity E 的启动模式设置为 SingleInstance,并且任务栈内不存在 Activity E 的实例,当启动 Activity E 时,会在创建一个新的任务栈,并且栈内只有 Activity E 一个实例

如果此时基于上面的任务栈,从 Activity D 中开启一个启动模式为 Standard 的 Activity F,那任务栈会发生什么样的变化呢?请看下图

此时 Activity E 依旧会独立的存在于自己的任务栈中,而新创建的 Activity F 将会和 Activity D 位于相同的任务栈的栈顶

如果启动模式为 SingleInstance 的 Activity E 已经独立存在于自己的任务栈中,此时再启动 Activity E, 则会复用已经创建的 Activity E 的实例,并且 Activity E 的 onNewIntent() 方法被调用

如果基于上面的任务栈,从 Activity E 中开启一个启动模式为 Standard 的 Activity F,那任务栈会发什么样的变化呢,请看下图

因为 singleInstance 的属性是禁止与其他 Activities 共享任务栈,所以启动模式为 SingleInstance 的 Activity 启动其他 Activity 时会默认带有 FLAG_ACTIVITY_NEW_TASK 属性。所以 Activity E 启动 Activity F 后,最后会存在三个任务栈,Activity F 会单独存在于一个任务栈中

参考