Android Activity栈关系解析

256 阅读3分钟

在 Android 系统中,这些类共同构成了 Activity 任务栈管理的核心架构。它们的关系可以类比为一栋大楼的管理体系,每个类负责不同层级的任务。以下是它们的详细解释和实际场景示例:

1. ActivityRecord(活动记录)

  • 是什么:代表一个 Activity 实例,存储 Activity 的元数据(如 Intent、启动模式、组件信息等)。

  • 职责:

    • 跟踪 Activity 的生命周期状态(如 RESUMED​、PAUSED​)。
    • 管理 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)。

层级关系总结

graph TD A[ActivityStackSupervisor] --> B[ActivityDisplay] B --> C[ActivityStack] C --> D[TaskRecord] D --> E[ActivityRecord]

实际协作流程示例

场景:用户在分屏模式下,左侧窗口运行微信,右侧窗口运行浏览器。

  1. ActivityStackSupervisor 创建两个 ActivityDisplay​(逻辑分屏视为两个虚拟屏幕)。

  2. 左侧 ActivityDisplay​ 创建一个 ActivityStack​,管理微信的 TaskRecord​(包含 MainActivity​ 和 ChatActivity​)。

  3. 右侧 ActivityDisplay​ 创建另一个 ActivityStack​,管理浏览器的 TaskRecord​(包含 HomePage​ 和 ArticlePage​)。

  4. 用户点击微信的返回键:

    • ​ActivityStackSupervisor​ 找到左侧 ActivityStack​ 的栈顶 ActivityRecord​(ChatActivity​)。
    • 销毁 ChatActivity​,回退到 MainActivity​。
  5. 用户拖拽浏览器窗口到左侧屏幕:

    • ​ActivityStackSupervisor​ 将浏览器的 TaskRecord​ 迁移到左侧 ActivityDisplay​ 的 ActivityStack​。

对开发者的意义

  1. 调试工具:

    • 通过 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}
      
  2. 启动模式与任务栈:

    • 使用 launchMode="singleTask"​ 时,系统会查找匹配的 TaskRecord​,若存在则复用,否则新建。
    • ​FLAG_ACTIVITY_NEW_TASK​ 强制在新的 TaskRecord​ 中启动 Activity。
  3. 多窗口适配:

    • 在分屏/自由窗口模式下,需处理 Configuration​ 变化(如屏幕尺寸、方向)。

总结

ActivityRecord 是砖块,TaskRecord 是房间,ActivityStack 是楼层,ActivityDisplay 是整栋楼,ActivityStackSupervisor 是物业总公司。