读官方文档 activities/tasks-and-back-stack总结
Launch Mode
Launch Mode 决定了 activity建立时候应如何与当前的 Task 栈联系起来。可以通过 manifest 或 在Intent 中添加 flags 来设置。后者的优先级更高。这两种方式是互相不包含的关系。
Using Manifest
- "standard" : 默认属性,activity 可被在同一 task 内被实例化多次,每个入口可属于不同 task , 一个 task 可以拥有多个入口。
- "singleTop" : 如果声明了此属性的activity在task栈顶,当有新intent要启动此activity时,会通过onNewIntent()来接收 intent 而不是新建实例。例:栈内有A-B-C-D,当有新 intent 指向D,"standard"是A-B-C-D-D,"singleTop"是A-B-C-D且D执行onNewIntent()且此时按下 back 键无法返回执行onNewIntent()前D的状态。
- "singleTask" : 新建 task 用以放入声明了此属性的activity,当已存在另一 task 里有要启动的此 activity 时,会通过onNewIntent()来接收 intent 而不是新建实例。此activity入口只能存在一个。尽管此activity在另一个task栈中,但是按下 back 键时还会返回它之前的activity。
-
"singleInstance" : 与"singleTask"类似,但声明了此属性的 activity 所在的task栈中只能有这唯一一个入口,它将要启动的 activities 都会放入新的task栈中。
例如浏览器应用的浏览器activity就是使用了"singleTask",在你的程序中启动浏览器时,被启动的浏览器activity并不是放入了你的程序的task中,而是转入了浏览器应用的task,并且当浏览器task存在时则调用onNewIntent()来接收你的程序发出的intent且该栈中还有它之前放入的activities,虽然体验是无缝的,但是task栈却发生了变化。
Using Intent Flags
- ** FLAG_ACTIVITY_NEW_TASK ** : same as "singleTask"
- ** FLAG_ACTIVITY_SINGLE_TOP ** : same as "singleTop"
- ** FLAG_ACTIVITY_CLEAR_TOP ** : 没有与之对应的launchMode,以此Flag打开Activity,会弹出栈中在目标activity之上的activities,并且以onNewIntent()来接收新intent。常与** FLAG_ACTIVITY_NEW_TASK **联合使用,用以定位在另外一个栈中的目标activity并将其置于能处理intent的栈顶。
android:taskAffinity
此属性描述了activity与所属task的关系,默认情况下同一应用的activities的此属性值是相同的(为包名),此值在两种情况下起作用:(1)启动activity时携带flag: FLAG_ACTIVITY_NEW_TASK,如果被启动的 activity的 taskAffinity 值与已存task栈中的已存activity相同,则不再分配新task而是启动这个已存在的task。(2)当android:allowTaskRepareting=true时
清除back stack
当task进入后台,系统会随时清理根activity以外栈中的所有activity,以下几个属性可以改变这一行为:
- alwaysRetainTaskState : 当根activity的此属性为true,那么系统会始终不清理该栈中的任何activity。
- ** clearTaskOnLaunch** : 当根activity的此属性为true,当task由后台进入前台时系统会始终清除根activity以外的activities,无论时间长短。
- ** finishOnTaskLaunch** : 与 clearTaskOnLaunch 类似,但它的作用域是在单个activity而不是task,当某activty的该属性为true,task由后台回到前台时该activity会被销毁,即使是根activity也可以。
Starting a task
以下代码片段将一个 activity 设置为了一个 task 的入口点。
<activity ... >
<intent-filter ... >
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
...
</activity>
文档中有这么一段: 想声明"singleTask" and "singleInstance"这两种用以启动新task的 launchMode 的 acitivty,必须也有ACTION_MAIN和CATEGORY_LAUNCHER 的 intent-filter。试想一下,如果启动以这两种 launchMode 启动了一个activity,并且在新栈中进行了一系列操作,此时按下home键,新栈进入了后台,却又没有为那声明这种intent-filter,用户就无法回到那个activity了。
一个额外收获
android导航栏的「“向上”按钮」与底部「返回键」的触发事件,根据情景的不同时不一样的,设计原则还是很巧妙的,详见 Navigation Guides。