ActivityStackSupervisor,ActivityStack,ActivityRecord,TaskRecord的关系

在学习activity的启动流程的时候,经常会碰到这几个类的身影,特别是在AMS中,ActivityStackSupervisor和ActivityStack对象的调用链经常切来换去的,比叫绕。

下面这篇文章就是梳理了下这几个类的关系。

ActivityRecord、TaskRecord、ActivityStack以及Activity启动模式详解

1.简介

先来张简单的关系图:

 

ActivityStack、TaskRecord、ActivityRecord关系图.png

  • 一个ActivityRecord对应一个Activity,保存了一个Activity的所有信息;但是一个Activity可能会有多个ActivityRecord,因为Activity可以被多次启动,这个主要取决于其启动模式。
  • 一个TaskRecord由一个或者多个ActivityRecord组成,这就是我们常说的任务栈,具有后进先出的特点。
  • ActivityStack则是用来管理TaskRecord的,包含了多个TaskRecord

下面进入详细的代码分析,本文源码基于android 27

2.代码分析

2.1 ActivityRecord

ActivityRecord,源码中的注释介绍:An entry in the history stack, representing an activity.
翻译:历史栈中的一个条目,代表一个activity。

    frameworks/base/services/core/java/com/android/server/am/ActivityRecord.java
    
    final class ActivityRecord extends ConfigurationContainer implements AppWindowContainerListener {

        final ActivityManagerService service; // owner
        final IApplicationToken.Stub appToken; // window manager token
        AppWindowContainerController mWindowContainerController;
        final ActivityInfo info; // all about me
        final ApplicationInfo appInfo; // information about activity's app
        
        //省略其他成员变量
        
        //ActivityRecord所在的TaskRecord
        private TaskRecord task;        // the task this is in.
        
        //构造方法,需要传递大量信息
        ActivityRecord(ActivityManagerService _service, ProcessRecord _caller, int _launchedFromPid,
                       int _launchedFromUid, String _launchedFromPackage, Intent _intent, String _resolvedType,
                       ActivityInfo aInfo, Configuration _configuration,
                       com.android.server.am.ActivityRecord _resultTo, String _resultWho, int _reqCode,
                       boolean _componentSpecified, boolean _rootVoiceInteraction,
                       ActivityStackSupervisor supervisor, ActivityOptions options,
                       com.android.server.am.ActivityRecord sourceRecord) {
        
        }
    }
  • 实际上,ActivityRecord中存在着大量的成员变量,包含了一个Activity的所有信息。
  • ActivityRecord中的成员变量task表示其所在的TaskRecord,由此可以看出:ActivityRecordTaskRecord建立了联系。

startActivity()时会创建一个ActivityRecord

    frameworks/base/services/core/java/com/android/server/am/ActivityStarter.java

    class ActivityStarter {
        private int startActivity(IApplicationThread caller, Intent intent, Intent ephemeralIntent,
                                  String resolvedType, ActivityInfo aInfo, ResolveInfo rInfo,
                                  IVoiceInteractionSession voiceSession, IVoiceInteractor voiceInteractor,
                                  IBinder resultTo, String resultWho, int requestCode, int callingPid, int callingUid,
                                  String callingPackage, int realCallingPid, int realCallingUid, int startFlags,
                                  ActivityOptions options, boolean ignoreTargetSecurity, boolean componentSpecified,
                                  com.android.server.am.ActivityRecord[] outActivity, TaskRecord inTask) {

            //其他代码略
            
            ActivityRecord r = new ActivityRecord(mService, callerApp, callingPid, callingUid,
                    callingPackage, intent, resolvedType, aInfo, mService.getGlobalConfiguration(),
                    resultRecord, resultWho, requestCode, componentSpecified, voiceSession != null,
                    mSupervisor, options, sourceRecord);
                    
            //其他代码略
        }
    }

2.2 TaskRecord

TaskRecord,内部维护一个ArrayList<ActivityRecord>用来保存ActivityRecord

    frameworks/base/services/core/java/com/android/server/am/TaskRecord.java
    
    final class TaskRecord extends ConfigurationContainer implements TaskWindowContainerListener {
        final int taskId;       //任务ID
        final ArrayList<ActivityRecord> mActivities;   //使用一个ArrayList来保存所有的ActivityRecord
        private ActivityStack mStack;   //TaskRecord所在的ActivityStack
        
        //构造方法
        TaskRecord(ActivityManagerService service, int _taskId, ActivityInfo info, Intent _intent,
                   IVoiceInteractionSession _voiceSession, IVoiceInteractor _voiceInteractor, int type) {
            
        }
        
        //添加Activity到顶部
        void addActivityToTop(com.android.server.am.ActivityRecord r) {
            addActivityAtIndex(mActivities.size(), r);
        }
        
        //添加Activity到指定的索引位置
        void addActivityAtIndex(int index, ActivityRecord r) {
            //...

            r.setTask(this);//为ActivityRecord设置TaskRecord,就是这里建立的联系

            //...
            
            index = Math.min(size, index);
            mActivities.add(index, r);//添加到mActivities
            
            //...
        }

        //其他代码略
    }
  • 可以看到TaskRecord中使用了一个ArrayList来保存所有的ActivityRecord
  • 同样,TaskRecord中的mStack表示其所在的ActivityStack

startActivity()时也会创建一个TaskRecord

    frameworks/base/services/core/java/com/android/server/am/ActivityStarter.java
    
    class ActivityStarter {

        private int setTaskFromReuseOrCreateNewTask(TaskRecord taskToAffiliate, int preferredLaunchStackId, ActivityStack topStack) {
            mTargetStack = computeStackFocus(mStartActivity, true, mLaunchBounds, mLaunchFlags, mOptions);

            if (mReuseTask == null) {
                //创建一个createTaskRecord,实际上是调用ActivityStack里面的createTaskRecord()方法,ActivityStack下面会讲到
                final TaskRecord task = mTargetStack.createTaskRecord(
                        mSupervisor.getNextTaskIdForUserLocked(mStartActivity.userId),
                        mNewTaskInfo != null ? mNewTaskInfo : mStartActivity.info,
                        mNewTaskIntent != null ? mNewTaskIntent : mIntent, mVoiceSession,
                        mVoiceInteractor, !mLaunchTaskBehind /* toTop */, mStartActivity.mActivityType);

                //其他代码略
            }
        }
    }

2.3 ActivityStack

ActivityStack,内部维护了一个ArrayList<TaskRecord>,用来管理TaskRecord

    frameworks/base/services/core/java/com/android/server/am/ActivityStack.java
    
    class ActivityStack<T extends StackWindowController> extends ConfigurationContainer implements StackWindowListener {

        private final ArrayList<TaskRecord> mTaskHistory = new ArrayList<>();//使用一个ArrayList来保存TaskRecord

        final int mStackId;

        protected final ActivityStackSupervisor mStackSupervisor;//持有一个ActivityStackSupervisor,所有的运行中的ActivityStacks都通过它来进行管理
        
        //构造方法
        ActivityStack(ActivityStackSupervisor.ActivityDisplay display, int stackId,
                      ActivityStackSupervisor supervisor, RecentTasks recentTasks, boolean onTop) {

        }
        
        TaskRecord createTaskRecord(int taskId, ActivityInfo info, Intent intent,
                                    IVoiceInteractionSession voiceSession, IVoiceInteractor voiceInteractor,
                                    boolean toTop, int type) {
                                    
            //创建一个task
            TaskRecord task = new TaskRecord(mService, taskId, info, intent, voiceSession, voiceInteractor, type);
            
            //将task添加到ActivityStack中去
            addTask(task, toTop, "createTaskRecord");

            //其他代码略

            return task;
        }
        
        //添加Task
        void addTask(final TaskRecord task, final boolean toTop, String reason) {

            addTask(task, toTop ? MAX_VALUE : 0, true /* schedulePictureInPictureModeChange */, reason);

            //其他代码略
        }

        //添加Task到指定位置
        void addTask(final TaskRecord task, int position, boolean schedulePictureInPictureModeChange,
                     String reason) {
            mTaskHistory.remove(task);//若存在,先移除
            
            //...
            
            mTaskHistory.add(position, task);//添加task到mTaskHistory
            task.setStack(this);//为TaskRecord设置ActivityStack

            //...
        }
        
        //其他代码略
    }
  • 可以看到ActivityStack使用了一个ArrayList来保存TaskRecord
  • 另外,ActivityStack中还持有ActivityStackSupervisor对象,这个是用来管理ActivityStacks的。

ActivityStack是由ActivityStackSupervisor来创建的,实际ActivityStackSupervisor就是用来管理ActivityStack的,继续看下面的ActivityStackSupervisor分析。

2.4 ActivityStackSupervisor

ActivityStackSupervisor,顾名思义,就是用来管理ActivityStack的。

    frameworks/base/services/core/java/com/android/server/am/ActivityStackSupervisor.java
    
    public class ActivityStackSupervisor extends ConfigurationContainer implements DisplayListener {

        ActivityStack mHomeStack;//管理的是Launcher相关的任务

        ActivityStack mFocusedStack;//管理非Launcher相关的任务
        
        //创建ActivityStack
        ActivityStack createStack(int stackId, ActivityStackSupervisor.ActivityDisplay display, boolean onTop) {
            switch (stackId) {
                case PINNED_STACK_ID:
                    //PinnedActivityStack是ActivityStack的子类
                    return new PinnedActivityStack(display, stackId, this, mRecentTasks, onTop);
                default:
                    //创建一个ActivityStack
                    return new ActivityStack(display, stackId, this, mRecentTasks, onTop);
            }
        }

    }
  • ActivityStackSupervisor内部有两个不同的ActivityStack对象:mHomeStackmFocusedStack,用来管理不同的任务。
  • ActivityStackSupervisor内部包含了创建ActivityStack对象的方法。

AMS初始化时会创建一个ActivityStackSupervisor对象。

2.5 总结

所以,实际上,他们的关系应该是这样的:

 

ActivityStack、TaskRecord、ActivityRecord、ActivityStackSupervisor关系图.png

3. 场景分析

下面通过启动Activity的代码来分析一下:

3.1 桌面

首先,我们看下处于桌面时的状态,运行命令:

adb shell dumpsys activity

结果如下:

ACTIVITY MANAGER ACTIVITIES (dumpsys activity activities)
Display #0 (activities from top to bottom):
  Stack #0:
  
  //中间省略其他...
  
    Task id #102
    
  //中间省略其他...
  
      TaskRecord{446ae9e #102 I=com.google.android.apps.nexuslauncher/.NexusLauncherActivity U=0 StackId=0 sz=1}
      Intent { act=android.intent.action.MAIN cat=[android.intent.category.HOME] flg=0x10000100 cmp=com.google.android.apps.nexuslauncher/.NexusLauncherActivity }
        Hist #0: ActivityRecord{54fa22 u0 com.google.android.apps.nexuslauncher/.NexusLauncherActivity t102}
          Intent { act=android.intent.action.MAIN cat=[android.intent.category.HOME] flg=0x10000100 cmp=com.google.android.apps.nexuslauncher/.NexusLauncherActivity }
          ProcessRecord{19c7c43 2203:com.google.android.apps.nexuslauncher/u0a22}

    Running activities (most recent first):
      TaskRecord{446ae9e #102 I=com.google.android.apps.nexuslauncher/.NexusLauncherActivity U=0 StackId=0 sz=1}
        Run #0: ActivityRecord{54fa22 u0 com.google.android.apps.nexuslauncher/.NexusLauncherActivity t102}

    mResumedActivity: ActivityRecord{54fa22 u0 com.google.android.apps.nexuslauncher/.NexusLauncherActivity t102}
    
//省略其他

实际上就是如下图所示的结构,这里的Stack #0就是ActivityStackSupervisor中的mHomeStackmHomeStack管理的是Launcher相关的任务。

桌面下的状态.png

 

3.2 从桌面启动一个Activity

从桌面启动一个APP,然后运行上面的命令,为了节省篇幅,这里和后面就不贴结果了,直接放图了。

从桌面点击图标启动一个AActivity,可以看到,会多了一个Stack #1,这个Stack #1就是ActivityStackSupervisor中的mFocusedStackmFocusedStack负责管理的是非Launcher相关的任务。同时也会创建一个新的ActivityRecordTaskRecordActivityRecord放到TaskRecord中,TaskRecord则放进mFocusedStack中。

启动A.png

3.3 默认模式从A启动B

然后,我们从AActivity中启动一个BActivity,可以看到会创建一个新的ActivityRecord然后放到已有的TaskRecord栈顶。

默认(standerd)模式启动B.png

 

3.4 从A启动B创建新栈

如果我们想启动的BActivity在一个新的栈中呢,我们可以用singleInstance的方式来启动BActivitysingleInstance后面也会讲到。这种方式会创建一个新的ActivityRecordTaskRecord,把ActivityRecord放到新的TaskRecord中去。

启动新Task.png

4. 流程分析

4.1 启动流程

这里对启动Activity过程中涉及到的ActivityStackTaskRecordActivityRecordActivityStackSupervisor进行简单的分析,实际上一张时序图就可以看明白了。相关的代码可以看上面的内容。

Activity启动UML类图.png

简单总结:
1.startActivity时首先会创建一个ActivityRecord
2.如果有需要,会创建一个TaskRecord,并把这个TaskRecord加入到ActivityStack中。
3.将ActivityRecord添加到TaskRecord的栈顶。

5. 启动模式

相信看完上面的介绍,现在再来看启动模式那是so easy了。

5.1 standerd

默认模式,每次启动Activity都会创建一个新的Activity实例。

比如:现在有个A Activity,我们在A上面启动B,再然后在B上面启动A,其过程如图所示:

standerd.png

5.2 singleTop

如果要启动的Activity已经在栈顶,则不会重新创建Activity,只会调用该该Activity的onNewIntent()方法。
如果要启动的Activity不在栈顶,则会重新创建该Activity的实例。

比如:现在有个A Activity,我们在A以standerd模式上面启动B,然后在B上面以singleTop模式启动A,其过程如图所示,这里会新创建一个A实例:

singleTop_A.png

如果在B上面以singleTop模式启动B的话,则不会重新创建B,只会调用onNewIntent()方法,其过程如图所示:

singleTop_B.png

 

5.3 singleTask

如果要启动的Activity已经存在于它想要归属的栈中,那么不会创建该Activity实例,将栈中位于该Activity上的所有的Activity出栈,同时该Activity的onNewIntent()方法会被调用。
如果要启动的Activity不存在于它想要归属的栈中,并且该栈存在,则会创建该Activity的实例。
如果要启动的Activity想要归属的栈不存在,则首先要创建一个新栈,然后创建该Activity实例并压入到新栈中。

比如:现在有个A Activity,我们在A以standerd模式上面启动B,然后在B上面以singleTask模式启动A,其过程如图所示:

singleTask.png

5.4 singleInstance

基本和singleTask一样,不同的是启动Activity时,首先要创建在一个新栈,然后创建该Activity实例并压入新栈中,新栈中只会存在这一个Activity实例。

比如:现在有个A Activity,我们在A以singleInstance模式上面启动B,其过程如图所示:

singleInstance.png

6.Intent的FLAG

另外,如果startActivity()时往Intent 中加入相应的标志来指定启动模式,这种方式的优先级会比在AndroidManifest中定义的优先级高;但是AndroidManifest中只能定义四种启动方式:standardsingleTopsingleTasksingleInstance,而Intentflag则有很多种。具体的可以看看文档,我们这里看看部分flag

  • FLAG_ACTIVITY_NEW_TASK :跟launchMode中的singleTask一样。
  • FLAG_ACTIVITY_SINGLE_TOP :跟launchMode中的singleTop一样。
  • FLAG_ACTIVITY_CLEAR_TOPlaunchMode中没有对应的值,如果要启动的Activity已经存在于栈中,则将所有位于它上面的Activity出栈。singleTask默认具有此标记位的效果。

 

----------------------------------------------------------------

看完后我的理解就是:

ActivityManagerService这个对象(AMS)是在SystemServer这个进程中创建的,整个Android系统就这么一个对象,该对象会持有一个ActivityStackSupervisor对象(简称ASS),这个ASS管理所有的ActivityStack(不太清楚在整个系统中会有多少个ActivityStack对象),每个ActivityStack对象又管理着多个TaskRecord,也就是我们常说的任务栈,相同属性的activity(在AMS中就是对应的ActivityRecord)被同一个TaskRecord管理着。这里的TaskRecord就是在我们在桌面调出多任务的时候显示出来的每个页面,这个页面显示的就是某个TaskRecord里最上面的一个activity的界面。

  • 2
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: 在 Android 系统中,ActivityRecordActivityStackTaskRecord 是三个重要的概念,它们分别表示活动记录、活动栈和任务记录。它们之间的关系如下: - ActivityRecord:表示一个活动(即一个界面),其中记录了该活动的状态、标志和其他信息。 - TaskRecord:表示一个任务,即一组相关的活动。一个任务可以包含多个活动,但是只有一个活动处于前台运行状态。 - ActivityStack:表示一个活动栈,即一组任务的集合。一个 ActivityStack 对象通常对应一个屏幕上的 Activity 栈。 接下来,我们来看一下它们之间是如何绑定的。 首先,ActivityRecord 对象是在 ActivityThread 的 handleLaunchActivity 方法中创建的,该方法会根据传入的参数创建一个新的 ActivityRecord 对象,并将其添加到 TaskRecord 中。在创建 ActivityRecord 对象时,会调用 TaskRecord 的 addActivityLocked 方法,将 ActivityRecord 对象添加到 TaskRecord 中。同时,ActivityRecord 对象中也会保存对 TaskRecord 对象的引用,以便后续使用。 接着,TaskRecord 对象是在 ActivityManagerService 的 startActivity 方法中创建的,该方法用于启动一个新的活动。当启动一个活动时,ActivityManagerService 会先查找是否存在该任务的 TaskRecord 对象,如果不存在则创建一个新的 TaskRecord 对象,并将其添加到 ActivityStack 中。此时,TaskRecord 对象中也会保存对 ActivityStack 对象的引用。 最后,ActivityStack 对象是在 ActivityStackSupervisor 的 createStack 方法中创建的,该方法用于创建一个新的 ActivityStack 对象。在创建 ActivityStack 对象时,会调用 ActivityStackSupervisor 的 addStackToWindowManager 方法,将 ActivityStack 对象添加到 WindowManagerService 中。同时,ActivityStack 对象中也会保存对 ActivityStackSupervisor 对象的引用。 综上所述,ActivityRecordActivityStackTaskRecord 之间的绑定关系是:ActivityRecord 对象保存对 TaskRecord 对象的引用,TaskRecord 对象保存对 ActivityStack 对象的引用,而 ActivityStack 对象保存对 ActivityStackSupervisor 对象的引用。 ### 回答2: ActivityRecordActivityStackTaskRecord是Android框架中用于管理Activity生命周期和任务栈的重要类。它们之间的绑定关系如下所述。 ActivityRecordActivity的代表,在内存中记录了每个Activity的信息,如任务栈的ID、Activity的状态、Intent等。ActivityStack是任务栈的管理类,用于管理任务栈中的ActivityTaskRecord是任务的代表,在内存中保存了一组相关联的ActivityActivityRecordTaskRecord之间的绑定关系是通过ActivityStack来建立的。在ActivityStack中,通过一个数组来保存TaskRecord,在TaskRecord中,通过一个链表来保存ActivityRecordActivityRecord中包含了TaskRecord的引用,用于表明该ActivityRecord属于哪个TaskRecord。 具体来说,在ActivityStack中,通过一系列操作将ActivityRecord添加到TaskRecord中。在启动一个新的Activity时,首先会查找目标Activity所在的TaskRecord。如果该TaskRecord不存在,则创建一个新的TaskRecord,并将ActivityRecord添加到该TaskRecord中。如果该TaskRecord已经存在,则将ActivityRecord添加到该TaskRecord中。在ActivityRecord中的task属性记录了所属TaskRecord的引用。 由于Activity可以有不同的启动模式(如standard、singleTop、singleTask、singleInstance等),所以在ActivityStack在添加ActivityRecordTaskRecord时需要考虑启动模式的不同。 总结起来,ActivityRecordActivityStackTaskRecord之间的绑定关系是通过ActivityRecord中的task属性和ActivityStack中的TaskRecord来建立的。ActivityRecord通过task属性指向所属的TaskRecord,而TaskRecord通过链表形式保存所属的ActivityRecord。这样就实现了在任务栈中管理Activity的目的。 ### 回答3: 在Android系统中,ActivityRecordActivityStackTaskRecord是三个重要的类,用于管理和调度Activity的生命周期和任务栈。下面是它们之间的绑定关系的源码分析。 1. ActivityRecord ActivityRecord是用于表示一个Activity实例的类,它包含了Activity的各种属性和状态。每个Activity实例都对应着一个唯一的ActivityRecord对象。在ActivityRecord类中,有一个成员变量称为task,用于保存当前Activity所属的TaskRecord对象。 2. TaskRecord TaskRecord是用于表示一个任务栈的类,它包含了一组ActivityRecord对象,形成了一个任务栈。每个TaskRecord对象都对应着一个唯一的任务栈。在TaskRecord类中,有一个成员变量称为stack,用于保存当前任务栈所属的ActivityStack对象。 3. ActivityStack ActivityStack是用于管理和调度任务栈的类,它包含了一组TaskRecord对象,形成了一个任务栈堆栈。在ActivityStack类中,有一个成员变量称为mTaskHistory,用于保存当前ActivityStack所维护的任务栈列表。 绑定过程如下: 1. 首先,当一个Activity启动时,Android会创建一个新的ActivityRecord对象,并给它分配一个唯一的taskId。 2. 接下来,ActivityStack会创建一个新的TaskRecord对象,并将该ActivityRecord对象加入到TaskRecord中。 3. 然后,TaskRecord对象会将自身加入到ActivityStack的mTaskHistory列表中,完成了TaskRecordActivityStack的绑定。 4. 同时,ActivityRecord对象会将自身的task成员变量指向所属的TaskRecord对象,完成了ActivityRecordTaskRecord的绑定。 5. 当一个Activity需要显示在屏幕上时,ActivityStack会将该Activity所在的TaskRecord对象置于栈顶,并将所属的ActivityStack对象置于前台。 通过以上步骤,完成了ActivityRecordActivityStackTaskRecord之间的绑定。这样,Android系统就能够根据TaskRecordActivityStack关系,对Activity启动、切换和销毁进行管理和调度。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值