我们知道,在默认情况下每当我们多次启动同一个Activity的时候,系统会创建多个实例并将他们一一放入任务栈中,当我们单击back键,会发现这些Activity会一一回退。多次启动同一个Activity的时候,系统会重复创建多个实例,这样不是很傻吗?所以安卓系统提供了启动模式launchMode来修改系统的默认行为。
目前有四种启动模式,相信所有的安卓开发者都是耳熟能详的:
standard、singleTop、singleTask和singleInstance
在理解这四种启动模式之前,我们需要理解taskAffinity概念
taskAffinity 任务吸引力
- taskAffinity是指Activity的归属,Activity与Task的吸附关系,也就是该Activity属于哪个Task。一般情况下在同一个应用中,启动的Activity都在同一个Task中,它们在该Task中度过自己的生命。
- 每个Activity都有taskAffinity属性,这个属性指出了它希望进入的Task。如果一个Activity没有显式的指明taskAffinity,那么它的这个属性就等于Application指明的taskAffinity,如果Application也没有指明,那么该taskAffinity的值就等于应用的包名。
- 我们可以通过在元素中增加taskAffinity属性来为某一个Activity指定单独的affinity。这个属性的值是一个字符串,可以指定为任意字符串,但是必须至少包含一个”.”,否则会报错。
- taskAffinity配置在Manifest中:
<activity
android:name="com.test.TestActivity"
android:configChanges="orientation|keyboard|keyboardHidden"
android:exported="true"
android:taskAffinity="com.test.TestActivity"
android:screenOrientation="portrait"/>
四种启动模式的含义
standard 标准模式
这也是系统的默认模式。每启动一个Activity都会重复创建一个新的实例,不管这个实例是否已经存在。在这种模式下,谁启动了这个Activity,那么这个Activity就运行在启动他的那个Activity所在的任务栈中。(由上面taskAffinity解释可知,如果一个Activity和他的Application都没有指定taskAffinity,那么他的taskAffinity默认为包名)
如果用ApplicationContext去启动standard模式的Activity,会发生如下报错
android.util.AndroidRuntimeException: Calling startActivity() from outside of an Activity context requires the FLAG_ACTIVITY_NEW_TASK flag.
这个报错我们很多同学可能都见过,这个因为standard模式的Activity会默认进入启动他的Activity所属的任务栈中。而非Activity类型的Context,并没有所谓的任务栈,这样就会出现crash。
解决这个问题的方法就是为启动Activity指定FLAG_ACTIVITY_NEW_TASK标志位,启动时为他创建一个新的任务栈。
singleTop 栈顶复用模式
在这种模式下,如果新的Activity已经位于任务栈的栈顶,那么此Activity不会被重新创建,同时他的onNewIntent方法会被回调,而onCreate、onStart不会被系统调用。
singleTask 栈内复用模式
在这种模式下,只要Activity在一个任务栈中存在,那么多次启动此Activity都不会创建新的实例,和singleTop一样,会回调onNewIntent。
这个过程的系统处理逻辑是:
当一个singleTask的Activity请求启动后,系统会先寻找是否存在A想要的任务栈(由taskAffinity指定)。
- 如果不存在,就创建一个任务栈,把此Activity放入栈顶。
- 如果已经存在想要的任务栈,这时分为两种情况:
如果栈中没有该Activity存在,则创建该Activity实例,并压入栈顶;
如果栈中有该Activity存在,则把该Activity调入栈顶,并把本来在他上面所有的Activity全部出栈,所以说singleTask模式默认带有clearTop效果。
singleInstance 单实例模式
这时一种增强的singleTask模式,他具有singleTask的全部特性。并且还有一个加强属性:他只能单独地位于一个任务栈中。
系统会为他启用一个新的栈结构,将Acitvity放置于这个新的栈结构中,并保证不再有其他Activity实例进入。
使用Intent的Flags指定启动模式
FLAG_ACTIVITY_NEW_TASK
单独的FLAG_ACTIVITY_NEW_TASK并不等价于启动模式 singleTask,它仅表示寻找Activity所需的任务栈压入(由taskAffinity指定)
FLAG_ACTIVITY_NEW_TASK与singleTask的关系
即使在FLAG_ACTIVITY_NEW_TASK | FLAG_ACTIVITY_CLEAR_TOP的情况下,AndroidManifest.xml中设置activity的启动模式为standard或singleTask时activity入栈方式是不一样的
- 当启动模式为standard时,如果activity所需的栈中已经存在该activity的实例了,那么这个实例连同它之上的activity都要出栈,然后再新建一个activity实例入栈。
- 当启动模式为singleTask时,如果activity所需的栈中已经存在该activity的实例了,那么系统会调用该实例的onNewIntent()方法,且只将该实例之上的activity出栈。
- 如果activity所需的栈中不存在该activity的实例,则不论启动模式为standard还是singleTask,都是新建activity实例直接入栈。
- AndroidManifest.xml中设置activity的启动模式为singleTask时,则不论是FLAG_ACTIVITY_NEW_TASK | FLAG_ACTIVITY_CLEAR_TOP还是只有FLAG_ACTIVITY_NEW_TASK的效果是一样的,因为singleTask模式中默认就带有FLAG_ACTIVITY_CLEAR_TOP标识。
FLAG_ACTIVITY_SINGLE_TOP
为Activity指定"singleTop"启动模式。
相信上文讲的很多细节,一些资深安卓程序员也有之前没有注意到的地方。
这边记录下来,方便我们查阅。