Android技术知识Android开发Android知识

Android Task相关属性详解

2017-08-13  本文已影响250人  carlos23

转自我2015年在博客园发布的文章Android Task 相关

前言

在日常开发过程中,只要涉及到activity,那么对task相关的东西总会或多或少的接触到,不过对task相关的一些配置的作用理解的还不是很透彻,官方文档在细节上说的也不够清楚,要透彻理解还是得自己写demo实践检验,所以便有了这篇总结。

task的概念

参考Tasks and Back Stack

查看设备当前task的方法

  1. View the Task’s activity stack
  2. Show Back Stack of Android

AndroidManifest中activity标签下和task有关的属性

谷歌现在对activity标签的文档出了中文版,阅读更加方便了。

taskAffinity

launchMode

launchMode有四种取值,与Intent里以FLAG_ACTIVITY_开头的flag结合,可以对activity的启动达到各种不同的效果。

standard

activity默认的启动模式,每次启动一个standard模式的activity时,都新建一个实例。

singleTop

当前task栈顶存在本activity的实例,直接使用该实例,调用该activity的onNewIntent(),否则新建一个activity的实例入栈。

singleTask

当启动一个singleTask模式的activity时,首先会检查是否存在与该activity的taskAffinity相同的task。

例如有四个activity叫A、B、C、D,其中A、B具有相同的affinity,现在taskA里有A、B,其中A和B是standard。

singleInstance

当启动一个singleInstance的activity时,首先会检查是否存在与该activity的taskAffinity相同的task。

例如有四个activity叫A、B、C、D,其中A、B具有相同的affinity,现在taskA里有A、B,其中A和B是standard

singleInstance使用场景举例:

  1. 闹钟应用的响铃页面
  2. 应用锁应用的锁屏页面

这两个页面从概念上理解,可以认作是独立的页面,跟其他页面无直接关系,应该单独放在一个task。

如果我们对这两个页面使用singleTask的话,就会产生一些逻辑问题,假设这两个页面现在都使用了用singleTask

如果响铃页面activity的affinity跟闹钟其他activity的affinity相同,那么先打开闹钟应用,产生taskA,再按home键打开其他应用,产生taskB,再等闹铃响起进入响铃页面,响铃页面的activity也会入栈taskA,同时taskA移动到taskB上面,这个时候关闭闹铃即关闭响铃页面后,返回的不是其他应用(taskB)而是按home键前闹钟应用里打开的最后一个页面,这跟预期的结果不一致。但如果响铃页面使用singleInstance,闹钟响起时,会产生taskC,关闭响铃页面后会返回其他应用(taskB)。

如果锁屏页面activity的affinity跟应用锁应用里其他activity的affinity不同,那么先打开应用锁应用,产生taskA,再按home键打开要上锁的应用,产生了taskB,然后后台服务检测要上锁,启动了锁屏页面,产生了taskC,整个task栈由底到顶依次是taskA->taskB->taskC,此时点击锁屏页面中的“忘记密码”的功能,打开忘记密码页面:

从上面的分析看来,从简单的角度还是直接设置这种独立的页面为singleInstance比较方便,不用考虑那么多问题。

allowTaskReparenting

此属性为true的activity被启动后,若有和此activity相同affinity的task转入前台,则此activity会从启动它的task移动到具有相同affinity的这个task。

例如,现在有两个应用分别为appA和appB,appA中有三个activity分别为activityA1、activityA2、activityA3,其中activityA1、activityA2的taskAffinity为taskA,activityA3的taskAffinity为taskB,appB中有一个activity为activityB1,其taskAffinity为taskB。所有activity都是standard模式。

启动appA,默认启动activityA1,再依次启动activityA2、activityA3,此时这三个activity都属于taskA。

按home键回到launcher,此时这三个activity扔都属于taskA。

alwaysRetainTaskState

如果用户离开一个task已经很久了,系统会在某个时刻清理掉这个task中除了根activity外所有的activity。当用户再次回到这个task,只有根activity被恢复。这样做是因为长期离开一个task,用户很有可能已经放弃了他之前所做的事情,转而要开始做新的事情,所以只保留根activity。

若根activity上的alwaysRetainTaskState为true,强制保留本task中的所有activity,即使过了很长时间,也不让系统清理task。

例如浏览器打开了很多个tab页,长时间不操作后也要保证再次回来时还是上次浏览的页面。

clearTaskOnLaunch

与alwaysRetainTaskState相反,若根activity上的clearTaskOnLaunch为true,不论何时用户再次从Launcher回到这个task时,除了根activity以外的其他activity都销毁。

finishOnTaskLaunch

此属性为true的activity,不论何时用户再次回到这个activity所属的task时,此activity会被销毁。此属性优先级优于allowTaskReparenting。

Intent中和task有关的部分flag

FLAG_ACTIVITY_NEW_TASK

和launchMode的属性值singleTask等效。如果一个Intent中包含此flag,尝试将要启动的activity放在一个新的task中,如果已经有一个task栈里存在目标activity的实例,将此task从后台调到前台来,调用已存在的activity实例的onNewIntent()方法。此flag不能用于startActivityForResult()。

FLAG_ACTIVITY_SINGLE_TOP

和launchMode的属性值singleTop等效。如果一个Intent中包含此属性,并且要启动的Activity就是当前的Activity(当前task栈顶activity),直接调用该activity的onNewIntent(),否则新建一个activity实例。

FLAG_ACTIVITY_CLEAR_TOP

如果一个Intent中包含此属性,并且当前task栈存存在目标activity的实例,清除该实例上面的所有的activity。
如果目标activity的launcherMode为standard,且Intent没有添加FLAG_ACTIVITY_SINGLE_TOP标记,则会销毁目标activity再重新创建,否则会重用该实例,调用onNewIntent()。

FLAG_ACTIVITY_CLEAR_WHEN_TASK_RESET

如果一个Intent中包含此属性,则它转向的那个Activity以及在那个Activity其上的所有Activity都会在task重置时被清除出task,这只发生在task重置的时候,而从Launcher中点击应用图标启动应用的时候会发生task重置(从Launcher启动应用会在Intent中附带一个FLAG_ACTIVITY_RESET_TASK_IF_NEEDED标记)。

参考资料

  1. Tasks and Back Stack
  2. Tasks and Back Stack中文翻译
  3. AndroidManifest中activity标签API Doc
  4. Intent API Doc
  5. Activity的task相关
上一篇 下一篇

猜你喜欢

热点阅读