Activity(三)栈管理

一、主要涉及4个类

1) ActivityRecord
源码注释:An entry in the history stack, representing an activity.
翻译一下:存在历史栈的一个实例,代表一个Activity。
2) TaskRecord
Activity栈,内部维护一个ArrayList<ActivityRecord>
3) ActivityStack
并不是一个Activity栈,真正意义上的Activity栈是TaskRecord,这个类是负责管理各个Activity栈,内部维护一个ArrayList<TaskRecord>
4) ActivityStackSupervisor
内部持有一个ActivityStack,而ActivityStack内部也持有ActivityStackSupervisor,相当于ActivityStack的辅助管理类

ActivityRecord
final class ActivityRecord {
    TaskRecord task;  // the task this is in.
    final IApplicationToken.Stub appToken;
    final int userId;
    int theme;
    int launchMode;
    ...
}

成员变量task表示自己所在的TaskRecord,这样要找到自己所在的TaskRecord就不必遍历查找了。

TaskRecord
final class TaskRecord {
    /** List of all activities in the task arranged in history order */
    final ArrayList<ActivityRecord> mActivities;

    /** Current stack */
    ActivityStack stack;
}

同样的道理,成员变量stack表示自己所在的ActivityStack

ActivityStack
final class ActivityStack {
    private ArrayList<TaskRecord> mTaskHistory = new ArrayList<>();

    /** Run all ActivityStacks through this */
    final ActivityStackSupervisor mStackSupervisor;

    ActivityStack(ActivityStackSupervisor.ActivityContainer activityContainer, RecentTasks recentTasks) {
        mStackSupervisor = activityContainer.getOuter();
        ...
    }
}
ActivityStackSupervisor
public final class ActivityStackSupervisor {
    private ActivityStack mFocusedStack;
}

二、场景解析

1、从桌面第一次启动App

startActivityLocked里构造一个ActivityRecord
新建一个TaskRecord,并存入mTaskHistory
ActivityRecord存入mActivities

final int startActivityUncheckedLocked(...) {
    final int startActivityUncheckedLocked(...) {
    if (reuseTask == null) {
        r.setTask(targetStack.createTaskRecord(...);
        ...
        targetStack.startActivityLocked(r, newTask, doResume, keepCurTransition, options);
        ...
    }
}
  1. TaskRecord存入mTaskHistory
 TaskRecord createTaskRecord(int taskId, ActivityInfo info, Intent intent,
            IVoiceInteractionSession voiceSession, IVoiceInteractor voiceInteractor,
            boolean toTop) {
        TaskRecord task = new TaskRecord(mService, taskId, info, intent, voiceSession,
                voiceInteractor);
        addTask(task, toTop, false);
        return task;
    }

    void addTask(final TaskRecord task, final boolean toTop, boolean moving) {
        task.stack = this;
        if (toTop) {
            insertTaskAtTop(task, null);
        } else {
            mTaskHistory.add(0, task);
            updateTaskMovement(task, false);
        }
        ...
    }

    private void insertTaskAtTop(TaskRecord task, ActivityRecord newActivity) {
        ...
        mTaskHistory.add(taskNdx, task);
        updateTaskMovement(task, true);
    }
  1. ActivityRecord存入mActivities
final void startActivityLocked(ActivityRecord r, boolean newTask, ...) {
    ...
    task = mTaskHistory.get(taskNdx);
    ...
    task.addActivityToTop(r);
}
void addActivityToTop(ActivityRecord r) {
    addActivityAtIndex(mActivities.size(), r);
}
void addActivityAtIndex(int index, ActivityRecord r) {
    ...
    mActivities.add(index, r);
    ...
}
2、App启动一个Activity

会不会新建一个TaskRecord取决于launchMode,默认的standard模式不会创建新的TaskRecord
构造一个ActivityRecord存入mActivities,与上面第二步一样

3、回退
/** @return true if this was the last activity in the task */
boolean removeActivity(ActivityRecord r) {
    mActivities.remove(r);
    ...
    if (mActivities.isEmpty()) {
        return !mReuseTask;
    }
    ...
    return false;
}

4、再次回退,回到桌面

从mActivities移除当前ActivityRecord与上面一样,只是当mActivities为空时,会触发mTaskHistory移除当前TaskRecord,如果mTaskHistory为空,则切换到桌面,给mStackSupervisor.mFocusedStack重新赋值

private void removeActivityFromHistoryLocked(ActivityRecord r, String reason) {
    ...
    final TaskRecord task = r.task;
    if (task != null && task.removeActivity(r)) {
        if (mStackSupervisor.isFrontStack(this) && task == topTask() &&
                task.isOverHomeStack()) {
            mStackSupervisor.moveHomeStackTaskToTop(task.getTaskToReturnTo(), reason);
        }
        removeTask(task, reason);
    }
}

void removeTask(TaskRecord task, String reason, boolean notMoving) {
    ...
    mTaskHistory.remove(task);
    ...
    if (mTaskHistory.isEmpty()) {
        final boolean notHomeStack = !isHomeStack();
        if (isOnHomeDisplay()) {
            String myReason = reason + " leftTaskHistoryEmpty";
            if (mFullscreen || !adjustFocusToNextVisibleStackLocked(null, myReason)) {
                mStackSupervisor.moveHomeStack(notHomeStack, myReason);
            }
        }
        ...
    }
    ...
    task.stack = null;
}

如果不是从Activity调用startActivity,那么目标Activity就不知道自己该属于哪个TaskRecord,所以得指定FLAG_ACTIVITY_NEW_TASK,就会新建一个TaskRecord

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 个人总结: 一个应用程序只有一个任务栈(Stack); 一个应用程序可以有多个Task(一个Task表现于回退栈,...
    画十阅读 1,744评论 0 3
  • 简介: 在android系统中,Home界面也就是Launcher的界面,Launcher它本身也是一个应用程序,...
    Memebox阅读 2,533评论 4 19
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 173,462评论 25 708
  • 看看日历,我们的生日又到了,只是:你我变成了爷奶,他她变成了姥爷和姥姥;只是:你我变成了旅游侠,他她变成了顾问和编...
    更新乱语阅读 155评论 0 0
  • 注:本文将从影片的色彩基调:“红与蓝”,以及它们所象征的“爱与悲”记录观影感受。 曾经,未见大鱼,就对它产生了无限...
    dk萍萍阅读 13,953评论 134 172