Android Activity栈关系解析
在 Android 系统中,这些类共同构成了 Activity 任务栈管理的核心架构。它们的关系可以类比为一栋大楼的管理体系,每个类负责不同层级的任务。以下是它们的详细解释和实际场景示例:
1. ActivityRecord(活动记录)
-
是什么:代表一个 Activity 实例,存储 Activity 的元数据(如 Intent、启动模式、组件信息等)。
-
职责:
- 跟踪 Activity 的生命周期状态(如
RESUMED
、PAUSED
)。 - 管理 Activity 的窗口(如界面层级、焦点状态)。
- 跟踪 Activity 的生命周期状态(如
-
示例:
// 当启动 MainActivity 时,系统会创建一个 ActivityRecord Intent intent = new Intent(context, MainActivity.class); startActivity(intent);
2. TaskRecord(任务记录)
-
是什么:代表一个 任务栈,用户视角中的一个“任务”(例如用户从桌面启动一个应用形成的任务)。
-
职责:
- 维护一组按顺序排列的
ActivityRecord
(后进先出,LIFO)。 - 处理任务栈的导航逻辑(如返回键回退)。
- 维护一组按顺序排列的
-
关键属性:
taskAffinity
:任务栈的“归属标识”,决定 Activity 应归属到哪个任务。rootActivity
:任务栈的根 Activity(第一个启动的 Activity)。
-
示例:
- 用户从桌面点击微信图标,启动
MainActivity
,生成一个TaskRecord
。 - 从
MainActivity
跳转到ChatActivity
,后者被压入同一任务栈。
- 用户从桌面点击微信图标,启动
3. ActivityStack(活动栈)
-
是什么:管理 一组相关的 TaskRecord,通常对应一个逻辑显示区域(如主屏幕、分屏窗口)。
-
职责:
- 控制 TaskRecord 的可见性和生命周期(如暂停后台栈中的 Activity)。
- 处理栈的焦点状态(如前台栈、后台栈)。
-
类型:
- 应用栈(Application Stack) :普通应用的任务栈。
- Home 栈:桌面和最近任务列表的栈。
- 特殊场景栈:如锁屏栈、语音交互栈。
-
示例:
- 分屏模式下,左侧窗口对应一个
ActivityStack
,右侧窗口对应另一个ActivityStack
。
- 分屏模式下,左侧窗口对应一个
4. ActivityDisplay(活动显示器)
-
是什么:管理 物理或虚拟显示设备(如手机主屏、外接显示器、分屏窗口)。
-
职责:
- 协调一个屏幕上的所有
ActivityStack
。 - 处理多屏交互(如窗口拖拽到另一个屏幕)。
- 协调一个屏幕上的所有
-
示例:
- 手机连接外接显示器时,系统创建两个
ActivityDisplay
,分别管理手机和显示器的任务栈。
- 手机连接外接显示器时,系统创建两个
5. ActivityStackSupervisor(活动栈监督者)
-
是什么:全局协调者,管理所有
ActivityDisplay
和ActivityStack
。 -
职责:
- 处理 Activity 的启动、切换、销毁等核心逻辑。
- 维护当前焦点栈(
mFocusedStack
)。 - 处理跨栈操作(如分屏模式下调整栈的层级)。
-
关键行为:
- 根据启动模式(如
singleTask
)和 Intent Flags(如FLAG_ACTIVITY_NEW_TASK
)决定 Activity 的归属栈。 - 处理返回键逻辑(回退栈顶 Activity)。
- 根据启动模式(如
层级关系总结
实际协作流程示例
场景:用户在分屏模式下,左侧窗口运行微信,右侧窗口运行浏览器。
-
ActivityStackSupervisor 创建两个
ActivityDisplay
(逻辑分屏视为两个虚拟屏幕)。 -
左侧
ActivityDisplay
创建一个ActivityStack
,管理微信的TaskRecord
(包含MainActivity
和ChatActivity
)。 -
右侧
ActivityDisplay
创建另一个ActivityStack
,管理浏览器的TaskRecord
(包含HomePage
和ArticlePage
)。 -
用户点击微信的返回键:
ActivityStackSupervisor
找到左侧ActivityStack
的栈顶ActivityRecord
(ChatActivity
)。- 销毁
ChatActivity
,回退到MainActivity
。
-
用户拖拽浏览器窗口到左侧屏幕:
ActivityStackSupervisor
将浏览器的TaskRecord
迁移到左侧ActivityDisplay
的ActivityStack
。
对开发者的意义
-
调试工具:
-
通过
adb shell dumpsys activity
查看完整的栈信息:adb shell dumpsys activity activities
-
输出示例:
Display #0 (手机主屏): Stack #0: type=standard, bounds=[0,0][1080,1920] Task #100: affinity=com.wechat, size=2 ActivityRecord{MainActivity} ActivityRecord{ChatActivity}
-
-
启动模式与任务栈:
- 使用
launchMode="singleTask"
时,系统会查找匹配的TaskRecord
,若存在则复用,否则新建。 FLAG_ACTIVITY_NEW_TASK
强制在新的TaskRecord
中启动 Activity。
- 使用
-
多窗口适配:
- 在分屏/自由窗口模式下,需处理
Configuration
变化(如屏幕尺寸、方向)。
- 在分屏/自由窗口模式下,需处理
总结
ActivityRecord 是砖块,TaskRecord 是房间,ActivityStack 是楼层,ActivityDisplay 是整栋楼,ActivityStackSupervisor 是物业总公司。