ActivityTask 和 Activity 栈

ActivityStack

ActivityStack 是一个管理类,用来管理系统所有 Activity 的各种状态,其内部维护了 TaskRecord 的列表,因此从 Activity 任务栈这一角度来说,ActivityStack 也可以理解为 Activity 堆栈。

它由 ActivityStackSupervisor 来进行管理的,而 ActivityStackSupervisor 在 AMS 中的构造方法中被创建。

image

ActivityStack 的实例类型

<pre class="md-fences md-end-block" lang="java" contenteditable="false" cid="n9" mdtype="fences" style="box-sizing: border-box; overflow: visible; font-family: monospace, monospace; font-size: 0.9rem; white-space: pre; line-height: 1.71429em; display: block; break-inside: avoid; text-align: left; background-image: inherit; background-position: inherit; background-size: inherit; background-repeat: inherit; background-attachment: inherit; background-origin: inherit; background-clip: inherit; background-color: rgb(218, 218, 218); position: relative !important; margin-bottom: 3em; padding-left: 1ch; padding-right: 1ch; margin-left: 2em; width: inherit; color: rgb(31, 9, 9); font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: 2; text-indent: 0px; text-transform: none; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px;">//com.android.server.am.ActivityStackSupervisor
public final class ActivityStackSupervisor implements DisplayListener {
...
ActivityStack mHomeStack;//存储 Launcher App 的所有 Activity
ActivityStack mFocusedStack;//当前正在接收输入或启动下一个 Activity 的所有 Activity
private ActivityStack mLastFocusedStack;//此前接收过输入的所有 Activity
...
}</pre>

ActivityRecord

ActivityRecord 用来记录一个 Activity 的所有信息。从 Activity 任务栈的角度来说,一个或多个 ActivityRecord 会组成一个 TaskRecord,TaskRecord 用来记录 Activity 的栈,而 ActivityStack 包含了一个或多个 TaskRecord。

<pre class="md-fences md-end-block" lang="java" contenteditable="false" cid="n13" mdtype="fences" style="box-sizing: border-box; overflow: visible; font-family: monospace, monospace; font-size: 0.9rem; white-space: pre; line-height: 1.71429em; display: block; break-inside: avoid; text-align: left; background-image: inherit; background-position: inherit; background-size: inherit; background-repeat: inherit; background-attachment: inherit; background-origin: inherit; background-clip: inherit; background-color: rgb(218, 218, 218); position: relative !important; margin-bottom: 3em; padding-left: 1ch; padding-right: 1ch; margin-left: 2em; width: inherit; color: rgb(31, 9, 9); font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: 2; text-indent: 0px; text-transform: none; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px;">
//在 ActivityStack 中定义了一些特殊状态的 Activity
ActivityRecord mPausingActivity = null;//正在暂停的Activity
ActivityRecord mLastPausedActivity = null;//上一个已经暂停的Activity
ActivityRecord mLastNoHistoryActivity = null;//最近一次没有历史记录的Activity
ActivityRecord mResumedActivity = null;//已经Resume的Activity
ActivityRecord mLastStartedActivity = null;//最近一次启动的Activity
ActivityRecord mTranslucentActivityWaiting = null;//传递给convertToTranslucent方法的最上层的Activity</pre>

ActivityState

<pre class="md-fences md-end-block" lang="java" contenteditable="false" cid="n15" mdtype="fences" style="box-sizing: border-box; overflow: visible; font-family: monospace, monospace; font-size: 0.9rem; white-space: pre; line-height: 1.71429em; display: block; break-inside: avoid; text-align: left; background-image: inherit; background-position: inherit; background-size: inherit; background-repeat: inherit; background-attachment: inherit; background-origin: inherit; background-clip: inherit; background-color: rgb(218, 218, 218); position: relative !important; margin-bottom: 3em; padding-left: 1ch; padding-right: 1ch; margin-left: 2em; width: inherit; color: rgb(31, 9, 9); font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: 2; text-indent: 0px; text-transform: none; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px;">
//通过枚举存储了 Activity 的所有的9种状态
enum ActivityState {
INITIALIZING,
RESUMED,
PAUSING,
PAUSED,
STOPPING,
STOPPED,
FINISHING,
DESTROYING,
DESTROYED
}</pre>

维护的 ArrayList

ActivityStack 中维护了很多 ArrayList,这些 ArrayList 中的元素类型主要有 ActivityRecord 和 TaskRecord,其中 TaskRecord 用来记录 Activity 的 Task。

ARRAYLIST 元素类型 说明
mTaskHistory TaskRecord 所有没有被销毁的 Task
mLRUActivities ActivityRecord 正在运行的 Activity,列表中的第一个条目是最近最少使用的元素
mNoAnimActivities ActivityRecord 不考虑转换动画的 Activity
mValidateAppTokens TaskGroup 用于与窗口管理器验证应用令牌

Activity 栈管理

Activity 是由任务栈来进行管理的,有了栈管理,我们可以对应用程序进行操作,应用可以复用自身应用中以及其他应用的 Activity,节省了资源。

比如我们使用一款社交应用,这个应用的联系人详情界面提供了联系人的邮箱,当我们点击邮箱时会跳到发送邮件的界面。社交应用和系统 Email 中的 Activity 是处于不同应用程序进程的,而有了栈管理,就可以把发送邮件界面放到社交应用中详情界面所在栈的栈顶,来做到跨进程操作。

image

Launch Mode

Launch Mode用于设定 Activity 的启动方式,无论是哪种启动方式,所启动的 Activity 都会位于 Activity 栈的栈顶。有以下四种:

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

  • singleTop:如果要启动的 Activity 已经在栈顶,则不会重新创建 Activity,同时该 Activity 的 onNewIntent 方法会被调用。如果要启动的 Activity 不在栈顶,则会重新创建该 Activity 的实例。

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

  • singleInstance:启动 Activity 时,首先要创建在一个新栈,然后创建该 Activity 实例并压入新栈中,新栈中只会存在这一个 Activity 实例。

Intent 的 FLAG

Intent 中定义了很多了 FLAG,其中有几个 FLAG 也可以设定 Activity 的启动方式,如果 Launch Mode 设定和 FLAG 设定的 Activity 的启动方式有冲突,则以 FLAG 设定的为准。

  • FLAG_ACTIVITY_SINGLE_TOP:和 Launch Mode 中的 singleTop 效果是一样的。

  • FLAG_ACTIVITY_NEW_TASK:和 Launch Mode 中的 singleTask 效果是一样的。这个名字比较奇怪要注意了。

  • FLAG_ACTIVITY_CLEAR_TOP:Launch Mode 中没有与此对应的模式,如果要启动的 Activity 已经存在于栈中,则将所有位于它上面的 Activity 出栈。singleTask 默认具有此标记位的效果。

除了这三个 FLAG,还有一些 FLAG 对我们分析栈管理有些帮助。

  • FLAG_ACTIVITY_NO_HISTORY:Activity 一旦退出,就不会存在于栈中。同样的,也可以在 AndroidManifest.xml 中设置 “android:noHistory”。

  • FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS:Activity 不会被放入到 “最近启动的 Activity” 列表中。

  • FLAG_ACTIVITY_BROUGHT_TO_FRONT:这个标志位通常不是由应用程序中的代码设置的,而是 Launch Mode 为 singleTask 时,由系统自动加上的。

  • FLAG_ACTIVITY_LAUNCHED_FROM_HISTORY:这个标志位通常不是由应用程序中的代码设置的,而是从历史记录中启动的(长按 Home 键调出)。

  • FLAG_ACTIVITY_MULTIPLE_TASK:需要和 FLAG_ACTIVITY_NEW_TASK 一同使用才有效果,系统会启动一个新的栈来容纳新启动的 Activity.

  • FLAG_ACTIVITY_CLEAR_TASK:需要和 FLAG_ACTIVITY_NEW_TASK 一同使用才有效果,用于清除与启动的 Activity 相关栈的所有其他 Activity。

计算Flag

根 Activity 启动时会调用 AMS 的 startActivity 方法,经过层层调用会调用 ActivityStarter 的 startActivityUnchecked 方法

<pre class="md-fences md-end-block" lang="java" contenteditable="false" cid="n100" mdtype="fences" style="box-sizing: border-box; overflow: visible; font-family: monospace, monospace; font-size: 0.9rem; white-space: pre; line-height: 1.71429em; display: block; break-inside: avoid; text-align: left; background-image: inherit; background-position: inherit; background-size: inherit; background-repeat: inherit; background-attachment: inherit; background-origin: inherit; background-clip: inherit; background-color: rgb(218, 218, 218); position: relative !important; margin-bottom: 3em; padding-left: 1ch; padding-right: 1ch; margin-left: 2em; width: inherit; color: rgb(31, 9, 9); font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: 2; text-indent: 0px; text-transform: none; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px;">
//com.android.server.am.ActivityStarter#startActivityUnchecked
private int startActivityUnchecked(final ActivityRecord r, ActivityRecord sourceRecord,
IVoiceInteractionSession voiceSession, IVoiceInteractor voiceInteractor,
int startFlags, boolean doResume, ActivityOptions options, TaskRecord inTask) {
setInitialState(r, options, inTask, doResume, startFlags, sourceRecord, voiceSession,
voiceInteractor);//重置各种配置再进行配置,如:ActivityRecord、Intent、TaskRecord 和 LaunchFlags(启动的 FLAG)等
computeLaunchingTaskFlags();//计算出启动的 FLAG,并将计算的值赋值给 mLaunchFlags
computeSourceStack();
mIntent.setFlags(mLaunchFlags);//将 mLaunchFlags 设置给 Intent,达到设定 Activity 的启动方式的目的
...
}
//com.android.server.am.ActivityStarter#computeLaunchingTaskFlags
private void computeLaunchingTaskFlags() {
...
if (mInTask == null) {// TaskRecord 类型的 mInTask 为 null 时,说明 Activity 要加入的栈不存在
if (mSourceRecord == null) {//初始 Activity(ActivityA 启动了 ActivityB,ActivityA 就是初始 Activity)为null
if ((mLaunchFlags & FLAG_ACTIVITY_NEW_TASK) == 0 && mInTask == null) {//需要创建一个新栈
Slog.w(TAG, "startActivity called from non-Activity context; forcing " +
"Intent.FLAG_ACTIVITY_NEW_TASK for: " + mIntent);
mLaunchFlags |= FLAG_ACTIVITY_NEW_TASK;
}
} else if (mSourceRecord.launchMode == LAUNCH_SINGLE_INSTANCE) {//如果 "初始 Activity" 所在的栈只允许有一个 Activity 实例,则需要创建一个新栈
mLaunchFlags |= FLAG_ACTIVITY_NEW_TASK;
} else if (mLaunchSingleInstance || mLaunchSingleTask) {//Launch Mode 设置了 singleTask 或 singleInstance,则也要创建一个新栈
mLaunchFlags |= FLAG_ACTIVITY_NEW_TASK;
}
}
}</pre>

taskAffinity

任务的亲近性,可以在 AndroidManifest.xml 设置 android:taskAffinity,用来指定 Activity 希望归属的栈, 默认情况下,同一个应用程序的所有的 Activity 都有着相同的 taskAffinity。

taskAffinity 在下面两种情况时会产生效果。

  1. taskAffinity 与 FLAG_ACTIVITY_NEW_TASK 或者 singleTask 配合。如果新启动 Activity 的 taskAffinity 和栈的 taskAffinity 相同(栈的 taskAffinity 取决于根 Activity 的 taskAffinity)则加入到该栈中。如果不同,就会创建新栈。

  2. taskAffinity 与 allowTaskReparenting 配合。如果 allowTaskReparenting 为 true,说明 Activity 具有转移的能力。拿此前的邮件为例,当社交应用启动了发送邮件的 Activity,此时发送邮件的 Activity 是和社交应用处于同一个栈中。如果发送邮件的 Activity 的 allowTaskReparenting 设置为 true,此后邮件程序所在的栈位于前台,这个时候发送邮件的 Activity 就会由社交应用的栈中转移到与它更亲近的邮件程序(taskAffinity 相同)所在的栈中。

taskAffinity的计算

<pre class="md-fences md-end-block" lang="java" contenteditable="false" cid="n114" mdtype="fences" style="box-sizing: border-box; overflow: visible; font-family: monospace, monospace; font-size: 0.9rem; white-space: pre; line-height: 1.71429em; display: block; break-inside: avoid; text-align: left; background-image: inherit; background-position: inherit; background-size: inherit; background-repeat: inherit; background-attachment: inherit; background-origin: inherit; background-clip: inherit; background-color: rgb(218, 218, 218); position: relative !important; margin-bottom: 3em; padding-left: 1ch; padding-right: 1ch; margin-left: 2em; width: inherit; color: rgb(31, 9, 9); font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: 2; text-indent: 0px; text-transform: none; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px;">
//com.android.server.am.ActivityStack#findTaskLocked
void findTaskLocked(ActivityRecord target, FindTaskResult result) {
...
for (int taskNdx = mTaskHistory.size() - 1; taskNdx >= 0; --taskNdx) {//遍历 mTaskHistory 列表,列表的元素为 TaskRecord,
用于存储没有被销毁的 Task
final TaskRecord task = mTaskHistory.get(taskNdx);//得到某一个 Task 的信息
...
else if (!isDocument && !taskIsDocument
&& result.r == null && task.canMatchRootAffinity()) {
if (task.rootAffinity.equals(target.taskAffinity)) {//将 Task 的 rootAffinity(初始的 taskAffinity)和目标 Activity 的 taskAffinity 做对比,如果相同,则将 FindTaskResult 的 matchedByRootAffinity 属性设置为 true,说明找到了匹配的 Task
if (DEBUG_TASKS) Slog.d(TAG_TASKS, "Found matching affinity candidate!");
result.r = r;
result.matchedByRootAffinity = true;
}
} else if (DEBUG_TASKS) Slog.d(TAG_TASKS, "Not a match: " + task);
}
}</pre>

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,324评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,303评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,192评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,555评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,569评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,566评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,927评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,583评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,827评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,590评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,669评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,365评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,941评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,928评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,159评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,880评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,399评论 2 342

推荐阅读更多精彩内容