启动模式(LaunchMode)
任务栈是一种“后进先出”的栈结构,在默认情况下,当我们多次启动同一个Activity的时候,系统会创建多个实例并把他们一一放入任务栈中,当我们点击back键时,这些Activity会一一回退(出栈),直到栈空为止,当栈中没有任何Activity的时候,系统就会回收这个任务栈。
那么问题来了,多次启动同一个Activity,系统重复创建多个实例,这不是有毛病吗?确实。所以Android提供了启动模式来修改系统的默认行为。
Android提供的四种启动模式分别为:standard、singleTop、singleTask、singleInstance。
-
standard
标准模式,也是系统的默认模式。每次启动一个Activity都会重新创建一个新的实例,不管这个实例是否已经存在。被创建的实例的生命周期符合典型情况下Activity的生命周期,也就是说,它的onCreate、onStart、onResume都会被调用。
在这种模式下,谁启动了这个Activity,那么这个Activity就运行在启动它的那个Activity所在的栈中。而使用非Activity的context来启动standard模式的Activity时,需要为待启动Activity指定FLAG_ACTIVITY_NEW_TASK标记位,否则就会报错,这是因为非Activity类型的context并没有所谓的任务栈,加上这个标记位后,启动时就会为它创建一个新的任务栈。 -
singleTop
栈顶复用模式,在这种模式下,如果新Activity已经位于任务栈的栈顶,那么此Activity不会被重新创建,同时它的onNewIntent方法会被回调,通过此方法的参数我们可以取出当前请求的信息。而且,这个Activity的onCreate、onStart都不会被系统调用,因为它并没有发生改变。
如果新Activity的实例已经存在但不是在栈顶,那么还是会重新创建这个Activity的实例。 -
singleTask
栈内复用模式,在这种模式下,只要Activity的实例在任务栈中,不管在不在栈顶,都不会被重新创建,而是把这个Activity调到栈顶,并且在它上面的Activity都会出栈,和singleTop一样,系统也会回调其onNewIntent方法。
这种模式存在一个任务栈匹配的过程,系统首先会寻找这个Activity需要的任务栈,如果如果这个任务栈不存在,则创建这个任务栈,然后把该Activity压入栈中,如果这个任务栈存在,就要看栈中是否有该Activity的实例,如果存在则将该Activity调到栈顶,如果不存在则创建该Activity的实例。 -
singleInstance
单实例模式,这是一种加强的singleTask模式,除了具有singleTask模式的所有特性外,还加强了一点,就是指定为此模式的Activity只能单独的位于一个任务栈中。
在singleTask启动模式中,提到了Activity所需要的任务栈,那么,什么是Activity需要的任务栈呢,如何指定呢?
默认情况下,所有Activity所需的任务栈的名字为应用的包名,当然,我们也可以通过TaskAffinity属性为每一个Activity指定任务栈的名字,这个指定的名字不能和包名相同,否则相当于没有指定。TaskAffinity属性主要和singleTask启动模式或者allowTaskReparenting属性配对使用,在其它情况下没有意义。另外,任务栈分为前台任务栈和后台任务栈,后台任务栈中的Activity位于暂停状态,用户可以通过切换将后台任务栈再次调到前台。
如何指定Activity的启动模式?
- 通过AndroidManifest.xml为Activity指定启动模式。
<activity android:name=".TestActivity"
android:launchMode="singleTask">
</activity>
- 通过在Intent中设置标记位来为Activity指定启动模式。
Intent intent = new Intent();
intent.setClass(this, TestActivity.class);
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
startActivity(intent);
这两种指定方式的区别在于:
- 优先级上,第二种方式要高于第一种,当两种同时存在时,以第二种为准。
- 限定范围上,第一种方式无法为Activity指定FLAG_ACTIVITY_CLEAR_TOP标识,第二种无法为Activity指定singleInstance模式。
标记位(Flags)
Activity的标记位有很多,这里主要介绍一些比较常用的标记位。
- FLAG_ACTIVITY_NEW_TASK
这个标记位的作用是为Activity指定“singleTask”启动模式,效果和在AndroidManifest.xml中指定“singleTask”模式相同。 - FLAG_ACTIVITY_SINGLE_TOP
这个标记位的作用是为Activity指定“singleTop”启动模式,效果和在AndroidManifest.xml中指定“singleTop”模式相同。 - FLAG_ACTIVITY_CLEAR_TOP
这个标记位表示的是具有此标记位的Activity,当它启动时,在同一个任务栈中所有位于它上面的Activity都要出栈。如果被启动的Activity采用standard模式启动,那么它连同它之上的Activity都要出栈,系统会创建新的Activity实例并压入栈顶。这个标记位与FLAG_ACTIVITY_NEW_TASK配合使用时,如果被启动的Activity已经存在,那么系统就会调用它的onNewIntent方法。 - FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS
具有这个标记位的Activity不会出现在历史Activity的列表中,当某些情况下我们不希望用户通过历史列表回到我们的Activity时可以使用这个标记位。它等同于在AndroidManifest.xml中指定Activity的属性android:excludeFromRecents="true"。
参考书籍《Android开发艺术探索》