美文网首页
Activity LaunchMode 测试与分析

Activity LaunchMode 测试与分析

作者: Jimmy_gjf | 来源:发表于2017-02-06 13:06 被阅读35次

在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没有关系
屏幕快照 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。 如下图:


屏幕快照 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

    MainActivity-SingleInstanceActivity.png
  • MainActivity-StandardActivity2: task156

    屏幕快照 2017-02-06 下午1.02.43.png

实验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
屏幕快照 2017-02-06 下午2.30.51.png
  • MainActivity-StandardActivity2: task159、task160
屏幕快照 2017-02-06 下午2.31.00.png

实验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
屏幕快照 2017-02-06 下午2.40.10.png

参考文献

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

相关文章

网友评论

      本文标题:Activity LaunchMode 测试与分析

      本文链接:https://www.haomeiwen.com/subject/dpoyittx.html