在adb shell环境下使用
dumpsys activity
命令查看activity所属stack、task信息。
总结
-
Task:
ActivityManagerService提供了一个ArrayList mHistory来管理所有的activity,activity在ActivityManagerService中的形式是ActivityRecord,task在ActivityManagerService中的形式为TaskRecord。我们可以简单的理解为mHistory包含很多TaskRecord,每个TaskRecord又包含很多ActivityRecord。Task是Activity的容器,每个Task都是由一系列的Activity组成的,每个应用对应一个或多个Task。
- 所有的ActivityRecord会被存储在mHistory管理
- 每个ActivityRecord会对应到一个TaskRecord,并且有着相同TaskRecord的ActivityRecord在mHistory中会处在连续的位置
- 同一个TaskRecord的Activity可能分别处于不同的进程中,每个Activity所处的进程跟task没有关系
- android:taskAffinity:
- 标记Activity所属task
- 默认值为应用包名
- 有效场景:
- LaunchMode=singleTask的Activity
- LaunchMode=singleInstance的Activity
- 使用flag=Intent.FLAG_ACTIVITY_NEW_TASK启动的Activity
- 在manifest中声明activity时taskAffinity即被指定,未特定设值Activity的taskAffinity属性均为系统默认值,即包名。
⚠️在进行不同task间的Activity拉起时,所有未设定taskAffinity值的Activity都属于系统默认task,与启动它的的task并无关联。
- singleTask:
- 设置了"singleTask"启动模式的Activity,它在启动的时候,会先在系统中查找属性值affinity等于它的属性值taskAffinity的任务存在;如果存在这样的任务,它就会在这个任务中启动,否则就会在新任务中启动。因此,如果我们想要设置了"singleTask"启动模式的Activity在新的任务中启动,就要为它设置一个独立的taskAffinity属性值。
⚠️如果未配置特定taskAffinity,直接调起launchMode为singleTask的Activity,并不会新建新的task,而是隶属于当前的task。
* 如果设置了"singleTask"启动模式的Activity不是在新的任务中启动时,它会在已有的任务中查看是否已经存在相应的Activity实例,如果存在,就会把位于这个Activity实例上面的Activity全部结束掉,即最终这个Activity实例会位于任务的堆栈顶端中。引自:解开Android应用程序组件Activity的"singleTask"之谜
官方文档:
无论 Activity 是在新任务中启动,还是在与启动 Activity 相同的任务中启动,用户按“返回”按钮始终会转到前一个 Activity。 但是,如果启动指定 singleTask 启动模式的 Activity,则当某后台任务中存在该 Activity 的实例时,整个任务都会转移到前台。此时,返回栈包括上移到堆栈顶部的任务中的所有 Activity。 如下图:
- singleTop:
- 如果在目标栈的顶部存在一个该Activity的实例,那么系统就会重用这个Activity的实例而不创建新的实例,并回调该Activity的onNewIntent(Intent intent)方法。
- 如果在目标栈的顶部没有该Activity的实例,系统将会在新建一个Activity实例。
- singleInstance:
- 不在乎是否设定特定taskAffinity,使用singleInstance启动的Activity都创建新的task。
官方解释:
* 该 Activity 始终是其任务唯一仅有的成员
* 由此 Activity 启动的任何 Activity 均在单独的任务中打开。
实验
实验1
manifest:
<activity android:name=".MainActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<activity android:name=".StandardActivity1"/>
<activity android:name=".StandardActivity2" />
<activity android:name=".SingleTopActivity" android:launchMode="singleTop"/>
<activity android:name=".SingleTaskActivity" android:launchMode="singleTask"/>
<activity android:name=".SingleInstanceActivity" android:launchMode="singleInstance"/>
onclick:
switch (v.getId()){
case R.id.standard1:
startActivity(new Intent(this, StandardActivity1.class));
break;
case R.id.standard2:
Intent intent = new Intent(this, StandardActivity2.class);
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
startActivity(intent);
break;
case R.id.single_task:
startActivity(new Intent(this, SingleTaskActivity.class));
break;
case R.id.single_top:
startActivity(new Intent(this, SingleTopActivity.class));
break;
case R.id.single_instance:
startActivity(new Intent(this, SingleInstanceActivity.class));
break;
default:
}
display:
-
MainActivity-SingleTaskActivity: task156
⚠️⚠️踩坑⚠️⚠️
踩:使用startActivityForResult方式启动flag为FLAG_ACTIVITY_NEW_TASK的Activity后无法收到result回调
解:当启动的Activity的flag为FLAG_ACTIVITY_NEW_TASK或者Activity的launchMode为singleTask时,那么新启动的activity将会与其caller断开依赖关系,这个关系主要是指result反馈,A-->B,如果A是通过startActivityForResult()请求启动的,并且requestCode >=0,无论B是否在新的Task中(如果B的taskAffinity值与A不同,则B在新的Task中,否则A、B均位于同一个Task中),B在finish的时候都不再向A反馈result,而是在启动过程中就会向A反馈一个RESULT_CANCELED。
-
MainActivity-SingleInstanceActivity: task156 、task157
-
MainActivity-StandardActivity2: task156
实验2
manifest:
<activity android:name=".StandardActivity1" android:taskAffinity="com.gjf.fun.standard1"/>
<activity android:name=".StandardActivity2" android:taskAffinity="com.gjf.fun.standard2"/>
onclick:
case R.id.standard1:
startActivity(new Intent(this, StandardActivity1.class));
break;
case R.id.standard2:
Intent intent = new Intent(this, StandardActivity2.class);
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
startActivity(intent);
break;
display:
- MainActivity-StandardActivity1: task159
- MainActivity-StandardActivity2: task159、task160
实验3
manifest:
<activity android:name=".SingleTaskActivity" android:launchMode="singleTask" android:taskAffinity="com.gjf.fun.singleTask"/>
onclick:
case R.id.single_task:
startActivity(new Intent(this, SingleTaskActivity.class));
break;
display:
- MainActivity-SingleTaskActivity2: task160、task161
参考文献
官方文档
android ActivityManagerService 源码分析----Activity管理(一)
这可能是目前最详细的安卓task, launchMode, intent flag测试分析与总结了
解开Android应用程序组件Activity的"singleTask"之谜
dumpsys命令用法