Activity的启动模式(LaunchMode)

时间:2021-08-14 16:46:42

1、Activity的LauchMode
首先说一下Acitiity为啥需要启动模式。我们知道,在默认情况下,当我们多次启动同一个Activity的时候,系统会创建多个实例并把它们一一放入任务栈中,当我们单击back键,会发现这些Activity会一一回退。任务栈是一种“先进后出”的栈结构。知道了Activity的默认启动模式以后,我们可能会发现一个问题:多次启动同一个Activity,系统重复创建多个实例,这样不是很low吗?所以Android在设计的时候不可能不考虑这个问题,所以它提供了启动模式来修改系统的默认行为。目前有四种启动模式:standard、singleTop、singleTask、singleInstance,下面先介绍各种启动模式的含义:
(1)standard:
标准模式,这也是系统的默认模式。每次启动一个Activity都会重新创建一个新的实例,不管这个实例存不存在。被创建的实例生命周期符合典型情况下的onCreate、onStart、onResume…
(2)singleTop:
栈顶复用模式。在这种模式下,如果新的Activity已经位于任务栈的栈顶,那么此Activity不会被重新创建,同时它的onNewIntent方法会被回调,通过此方法的参数我们可以取出当前请求的信息。此时,这个Activity的onCreate、onStart不会被系统调用。当然新的Activity如果不是位于栈顶,就那么仍然会重新创建。举个例子,假设摸钱站内的情况为ABCD,其中ABCD四个Activity,D位于栈顶,A位于栈底,这个时候假设要再次启动D,如果D的启动模式为singleTop,那么栈内的情况仍然为ABCD,如果D为的启动模式为standard,那么,由于D被重新创建,那么栈内的情况为ABCDD。
(3)singleTask:
栈内复用模式。这是一种单例模式,在这种模式下,只要一个栈存在,那么多次启动此Activity都不会重新创建实例,同样也会回调其onNewIntent。例如:
存在当期任务栈为ADBC,此时D的启动模式为singleTask此时再次启动D的时候,D不会重建,系统会把D切到栈顶,同时由于singleTask默认具有clearTop的效果,会导致栈内所有D上面的Activity全部出栈,于是最终的情况为AD。
(4)singleInstance:
单实例模式。这是一种加强的singleTask模式,它除了具有singleTask模式的所有特性之外,还加强了一点,那就是具有此种模式的Activity只能单独的位于一个任务栈中。

2、任务栈:
在Activity的启动模式中“任务栈”多次被提起:究竟何为任务栈呢?默认情况下,所有Activity所需的任务栈的名字为应用包名,一个应用中所有的Activity都是同一个任务栈中。另外这里有一个参数:TaskAffinity,可以翻译为人物相关性。这个参数表示了一个Activity所需要的任务栈的名字,当然,我们可以给每个Activity都单独制定TaskAffinity属性,这个属性值必须不能和包名相同,否则就相当于没有指定。TaskAffinity属性主要和singleTask启动模式或者alowTaskPeparenting属性配对使用,在其他情况下没有意义。

3、Activity的Flags
Activity的Flags有很多,这里主要分析一下比较常用的标记。标记位的作用有很多,有的标记位可以设定Activity的启动模式,比如,FLAG_ACTIVITY_NEW_TASK 和 F LAG_ACTIVITY_SINGLE_TOP等;还有的标记可以影响到Activity的运行状态,比如FLAG_ACTIVITY_CLEAR_TOP和FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS等。下面主要介绍几个比较常用的标记位,剩下的标记位读者可自行查看官方文档。
FLAG_ACTIVITY_NEW_TASK
这个标记位的作用是为Activity指定“singleTask”启动模式,其效果和在XML中指定该启动模式作用相同。
FLAG_ACTIVITY_SINGLE_TOP
这个标记位的作用是为Activity指定“singleTop”启动模式,其效果和在XML中指定该启动模式作用相同。
FLAG_ACTIVITY_CLEAR_TOP
具有此标记位的Activity,当它启动时,在同一个任务栈中所有位于它上面的Activity都要出栈。这个标记位一般会和singleTask启动模式一起出现,从上面的分析我们可以知道,singleTask启动模式默认就具有此标记位的效果。
FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS
具有这个标记的Activity不会出现在历史Activity的列表中,当某些情况下我们不希望用户通过历史列表回到我们的Activity的时候这个标记比较有用。它等同于在XML中指定Activity的属性 android:excludeFromRecents = “true”。

4、代码实现

一种是在androidmainfest.xml中配置:

<activity
android:name="com.itcode.taskstack.SecondActivity"
android:label="@string/_second"
android:launchMode="singleTop">

</activity>

另一种情况是通过在Intent中设置标志位来为Activity指定启动模式,比如:

    Intent intent = new Intent();
intent.setClass(MainActivity.this, LoginActivity.class);
intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TASK);
startActivity(intent);

这两种方式都可以为Activity指定启动模式,但是二者还是有区别的。
首先,优先级上,第二种方式的优先级要高于第一种,当两种同时存在是,以第二种为准。
其次,上述两种方式在限定范围上有所不同,
第一种方式无法直接为Activity设定FLAG_ACTIVITY_CLEAR_TOP标识,
第二种方式无法为Activity指定singleInstance模式。