Android 中的 Activity 有几种比较重要的启动模式,Standard\SingleTop\SingleTask\SingleInstance , 每一种启动模式有不同的使用场景, 网上也有许多分析这个的文章, 这里我以 Demo 的模式, 从 Activity 栈的角度来展示不同启动模式下的 Activity 的行为.
Activity 栈是一个先进后出的数据结构, 各位可以关注在每一步操作之后, 栈内容那一栏 , 可以更好地帮助理解不同的启动模式.
Demo 比较简单, 我也放到了 Github 上 , https://github.com/Gracker/AndroidLaunchModeTest , 有兴趣的可以自己跑一下 , 看看结果 , 只需要修改 StandardActivity 里面的跳转 Activity 就可以了.
Standard 标准模式 1 android: launchMode="standard"
最基本的模式,每次启动都会创建一个新的 Activity
模式 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 // 1. 启动 Activity MainActivity //栈内容 com.example.launchmodetest/.MainActivity ------------------------------------------------------------------- // 2. 启动 StandardActivity MainActivity -> StandardActivity //栈内容 com.example.launchmodetest/.StandardActivity com.example.launchmodetest/.MainActivity ------------------------------------------------------------------- // 3. 启动 StandardActivity MainActivity -> StandardActivity -> StandardActivity //栈内容 com.example.launchmodetest/.StandardActivity com.example.launchmodetest/.StandardActivity com.example.launchmodetest/.MainActivity
SingleTop 栈顶复用模式 1 android: launchMode="singleTop"
如果当前 Activity 已经在栈顶,那么其 onNewIntent 会被调用;否则会重新创建 Activity
测试1 : SingleTopActivity 不在栈顶 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 MainActivity com.example .launchmodetest/.MainActivity ------------------------------------------------------------------- MainActivity -> StandardActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.MainActivity ------------------------------------------------------------------- MainActivity -> StandardActivity -> SingleTopActivity com.example .launchmodetest/.SingleTopActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.MainActivity ------------------------------------------------------------------- MainActivity -> StandardActivity -> SingleTopActivity -> StandardActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.SingleTopActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.MainActivity ------------------------------------------------------------------- MainActivity -> StandardActivity -> SingleTopActivity -> StandardActivity -> SingleTopActivity com.example .launchmodetest/.SingleTopActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.SingleTopActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.MainActivity
测试2 : SingleTopActivity 在栈顶 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 MainActivity com.example .launchmodetest/.MainActivity ------------------------------------------------------------------- MainActivity -> StandardActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.MainActivity ------------------------------------------------------------------- MainActivity -> StandardActivity -> SingleTopActivity com.example .launchmodetest/.SingleTopActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.MainActivity ------------------------------------------------------------------- MainActivity -> StandardActivity -> SingleTopActivity -> SingleTopActivity com.example .launchmodetest/.SingleTopActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.MainActivity
SingleTask 栈内复用模式 1 android: launchMode="singleTask"
如果不加 Affinity , 那么 SingleTask 标记的 Activity 创建还是在当前的 Task 中
SingleTask 标记的 Activity 是栈内复用模式,如果当前 Task 内没有这个 Activity,那么创建新的 Activity,如果当前 Task 内有这个 Activity,不管他在 Task 的哪个位置,都会直接复用这个 Activity (收到 onNewIntent)
如果栈内复用,那么会 Clear Task 中这个 Activity 上面的其他的 Activity
测试1:SingleTask(Without Affinity) 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 MainActivity com.example .launchmodetest/.MainActivity ------------------------------------------------------------------- MainActivity -> StandardActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.MainActivity ------------------------------------------------------------------- MainActivity -> StandardActivity -> SingleTaskActivity com.example .launchmodetest/.SingleTaskActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.MainActivity ------------------------------------------------------------------- MainActivity -> StandardActivity -> SingleTaskActivity -> StandardActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.SingleTaskActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.MainActivity ------------------------------------------------------------------- MainActivity -> StandardActivity -> SingleTaskActivity -> StandardActivity -> SingleTaskActivity com.example .launchmodetest/.SingleTaskActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.MainActivity
测试2:SingleTask(WithAffinity) 在 Manifest 中设置了 android:taskAffinity=”” 之后,启动 SingleTask 会启动一个新的 Task
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 MainActivity com.example .launchmodetest/.MainActivity ------------------------------------------------------------------- MainActivity -> StandardActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.MainActivity ------------------------------------------------------------------- MainActivity -> StandardActivity -> SingleTaskWithAffinity com.example .launchmodetest/.SingleTaskWithAffinity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.MainActivity ------------------------------------------------------------------- MainActivity -> StandardActivity -> SingleTaskWithAffinity -> StandardActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.SingleTaskWithAffinity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.MainActivity ------------------------------------------------------------------- MainActivity -> StandardActivity -> SingleTaskWithAffinity -> StandardActivity -> SingleTaskWithAffinity com.example .launchmodetest/.SingleTaskWithAffinity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.MainActivity -------------------------------------------------------------------
总结
与 SingleTask 相比, SingleTaskWithAffinity 会创建新的 Stack
在 SingleTaskWithAffinity 启动 StandardActivity , 这个 StandardActivity 与 SingleTaskWithAffinity 在同一个栈
在栈 0 里面再启动 SingleTaskWithAffinity ,不会创建新的 Task
多任务里面会出现 SingleTaskWithAffinity
SingleInstance 单实例模式 1 android: launchMode="singleInstance"
单示例模式顾名思义,启动时,无论从哪里启动都会给 A 创建一个唯一的任务栈,后续的创建都不会再创建新的 A,除非 A 被销毁了
测试1:SingleInstance (Without Affinity) 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 taskAffinity=com.example .launchmodetest MainActivity com.example .launchmodetest/.MainActivity ------------------------------------------------------------------- MainActivity -> StandardActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.MainActivity ------------------------------------------------------------------- MainActivity -> StandardActivity -> SingleInstanceActivity com.example .launchmodetest/.SingleInstanceActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.MainActivity ------------------------------------------------------------------- MainActivity -> StandardActivity -> SingleInstanceActivity -> StandardActivity com.example .launchmodetest/.SingleInstanceActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.MainActivity ------------------------------------------------------------------- MainActivity -> StandardActivity -> SingleInstanceActivity -> StandardActivity -> SingleInstanceActivity com.example .launchmodetest/.SingleInstanceActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.MainActivity ------------------------------------------------------------------- MainActivity -> StandardActivity -> SingleInstanceActivity -> StandardActivity -> SingleInstanceActivity -> StandardActivity com.example .launchmodetest/.SingleInstanceActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.MainActivity ------------------------------------------------------------------- MainActivity -> StandardActivity -> SingleInstanceActivity -> StandardActivity -> SingleInstanceActivity -> StandardActivity -> SingleInstanceActivity com.example .launchmodetest/.SingleInstanceActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.MainActivity
总结1
SingleInstanceActivity 会创建新的 Task ,但是不会在多任务中出现
SingleInstanceActivity 是全局唯一的,如果复用,其 onNewIntent 会被调用
SingleInstanceActivity 启动新的 Activity,新的 Activity 不会在当前的 Task 里面,而是会回到上一个 Task 里面
测试2: SingleInstance (With Affinity) 在 Manifest 中设置了 android:taskAffinity=”” 之后,启动 SingleInstanceActivity 会出现在多任务中 ,其余的表现与没有设置 Affinity 一致
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 MainActivity com.example .launchmodetest/.MainActivity ------------------------------------------------------------------- MainActivity -> StandardActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.MainActivity ------------------------------------------------------------------- MainActivity -> StandardActivity -> SingleInstanceWithAffinityActivity com.example .launchmodetest/.SingleInstanceWithAffinityActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.MainActivity ------------------------------------------------------------------- MainActivity -> StandardActivity -> SingleInstanceWithAffinityActivity -> StandardActivity com.example .launchmodetest/.SingleInstanceWithAffinityActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.MainActivity ------------------------------------------------------------------- MainActivity -> StandardActivity -> SingleInstanceWithAffinityActivity -> StandardActivity -> SingleInstanceWithAffinityActivity com.example .launchmodetest/.SingleInstanceWithAffinityActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.MainActivity ------------------------------------------------------------------- MainActivity -> StandardActivity -> SingleInstanceWithAffinityActivity -> StandardActivity -> SingleInstanceWithAffinityActivity -> StandardActivity com.example .launchmodetest/.SingleInstanceWithAffinityActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.MainActivity ------------------------------------------------------------------- MainActivity -> StandardActivity -> SingleInstanceWithAffinityActivity -> StandardActivity -> SingleInstanceWithAffinityActivity -> StandardActivity -> SingleInstanceWithAffinityActivity com.example .launchmodetest/.SingleInstanceWithAffinityActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.StandardActivity com.example .launchmodetest/.MainActivity
一些概念 TaskAffinity
与 Activity 有着亲和关系的任务。从概念上讲,具有相同亲和关系的 Activity 归属同一Task(从用户的角度来看,则是归属同一“ Application ”)。 Task 的亲和关系由其根 Activity 的亲和关系确定。
亲和关系确定两件事 - Activity 更改到的父项 Task(请参阅 allowTaskReparenting 属性)和通过 FLAG_ACTIVITY_NEW_TASK 标志启动 Activity 时将用来容纳它的 Task。 默认情况下,应用中的所有 Activity 都具有相同的亲和关系。您可以设置该属性来以不同方式组合它们,甚至可以将在不同应用中定义的 Activity 置于同一 Task 内。 要指定 Activity 与任何 Task 均无亲和关系,请将其设置为空字符串。
如果未设置该属性,则 Activity 继承为应用设置的亲和关系(请参阅 元素的 taskAffinity 属性)。 应用默认亲和关系的名称是 元素设置的软件包名称。
ActivityRecord、TaskRecord、ActivityStack 之间的关系
一个 ActivityRecord 对应一个 Activity 实例,保存了一个 Activity 的所有信息 ; 但是一个 Activity可能会有多个 ActivityRecord ,因为 Activity 可以被多次启动,这个主要取决于其启动模式。
一个 TaskRecord 由一个或者多个 ActivityRecord 组成,这就是我们常说的任务栈,具有后进先出的特点
ActivityStack 则是用来管理 TaskRecord 的,包含了多个 TaskRecord
(From http://gityuan.com/2017/06/11/activity_record/)
一般地,对于没有分屏功能以及虚拟屏的情况下,ActivityStackSupervisor 与ActivityDisplay 都是系统唯一;
ActivityDisplay 主要有 Home Stack 、 App Stack、Recents Stack 这三个栈;
每个 ActivityStack 中可以有若干个 TaskRecord 对象;
每个 TaskRecord 包含如果若干个 ActivityRecord 对象;
每个 ActivityRecord记 录一个 Activity 信息。
下面是一个 dump 的例子,可以看到当前手机的 ActivityRecord、TaskRecord、ActivityStack (adb shell dumpsys activity containers)
Activity 的几种类型 1 2 3 4 5 6 7 8 9 10 public static final int ACTIVITY_TYPE_UNDEFINED = 0 ;public static final int ACTIVITY_TYPE_STANDARD = 1 ;public static final int ACTIVITY_TYPE_HOME = 2 ;public static final int ACTIVITY_TYPE_RECENTS = 3 ;public static final int ACTIVITY_TYPE_ASSISTANT = 4 ;
如果觉得文章有帮助, 欢迎分享到社交网站 , 希望能帮到大家.
关于我 && 博客 下面是个人的介绍和相关的链接,期望与同行的各位多多交流,三人行,则必有我师!
博主个人介绍 :里面有个人的微信和微信群链接。
本博客内容导航 :个人博客内容的一个导航。
个人整理和搜集的优秀博客文章 - Android 性能优化必知必会 :欢迎大家自荐和推荐 (微信私聊即可)
Android性能优化知识星球 : 欢迎加入,多谢支持~
一个人可以走的更快 , 一群人可以走的更远