Activity LaunchMode 测试与分析

在adb shell环境下使用dumpsys activity命令查看activity所属stack、task信息。

总结

  1. Task:

    ActivityManagerService提供了一个ArrayList mHistory来管理所有的activity,activity在ActivityManagerService中的形式是ActivityRecord,task在ActivityManagerService中的形式为TaskRecord。我们可以简单的理解为mHistory包含很多TaskRecord,每个TaskRecord又包含很多ActivityRecord。Task是Activity的容器,每个Task都是由一系列的Activity组成的,每个应用对应一个或多个Task。

引自:android ActivityManagerService 源码分析----Activity管理(一)

  • 所有的ActivityRecord会被存储在mHistory管理
  • 每个ActivityRecord会对应到一个TaskRecord,并且有着相同TaskRecord的ActivityRecord在mHistory中会处在连续的位置
  • 同一个TaskRecord的Activity可能分别处于不同的进程中,每个Activity所处的进程跟task没有关系
Activity LaunchMode 测试与分析_第1张图片
屏幕快照 2017-02-07 下午12.07.42.png
  1. 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并无关联。

  2. 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。 如下图:


Activity LaunchMode 测试与分析_第2张图片
  1. singleTop:
    • 如果在目标栈的顶部存在一个该Activity的实例,那么系统就会重用这个Activity的实例而不创建新的实例,并回调该Activity的onNewIntent(Intent intent)方法。
    • 如果在目标栈的顶部没有该Activity的实例,系统将会在新建一个Activity实例。
  2. singleInstance:
    • 不在乎是否设定特定taskAffinity,使用singleInstance启动的Activity都创建新的task。

    官方解释:
    * 该 Activity 始终是其任务唯一仅有的成员
    * 由此 Activity 启动的任何 Activity 均在单独的任务中打开。

实验

实验1

manifest:


   
        
        
   






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

    Activity LaunchMode 测试与分析_第3张图片
    屏幕快照 2017-02-06 下午1.04.22.png

⚠️⚠️踩坑⚠️⚠️
踩:使用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

    Activity LaunchMode 测试与分析_第4张图片
    MainActivity-SingleInstanceActivity.png
  • MainActivity-StandardActivity2: task156

    Activity LaunchMode 测试与分析_第5张图片
    屏幕快照 2017-02-06 下午1.02.43.png

实验2

manifest:



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
Activity LaunchMode 测试与分析_第6张图片
屏幕快照 2017-02-06 下午2.30.51.png
  • MainActivity-StandardActivity2: task159、task160
Activity LaunchMode 测试与分析_第7张图片
屏幕快照 2017-02-06 下午2.31.00.png

实验3

manifest:


onclick:

case R.id.single_task:
     startActivity(new Intent(this, SingleTaskActivity.class));
     break;

display:

  • MainActivity-SingleTaskActivity2: task160、task161
Activity LaunchMode 测试与分析_第8张图片
屏幕快照 2017-02-06 下午2.40.10.png

参考文献

官方文档
android ActivityManagerService 源码分析----Activity管理(一)
这可能是目前最详细的安卓task, launchMode, intent flag测试分析与总结了
解开Android应用程序组件Activity的"singleTask"之谜
dumpsys命令用法

你可能感兴趣的:(Activity LaunchMode 测试与分析)