我们知道Activity的启动模式有四种分别为:standard(标准模式)、singleTop(栈顶复用模式)、singleTask(栈内复用模式)和singleInstance(单一实例模式),在安卓系统中Activity通过任务栈的方式管理Activity,不同的启动模式实际上对应的是对任务栈的不同操作,要想搞明白四种启动模式有什么不同,我们先来了解一下任务栈
一 任务栈
- 安卓系统通过任务栈来管理Activity
- 栈具有后进先出的特性,每次打开一个Activity都会进行压栈操作,每次销毁一个Activity都会进行出栈操作
- 任何一个任务栈(task)都一定有一个taskAffinity值,该值与该任务栈栈底的Acitivity的taskAffinity值相同
- 每个Activity都必须有一个taskAffinity值,如果没有设置则该值等于application中的taskAffinity值,如果application中也没有设置taskAffinity,则该值为包名
二 四种启动模式
Activity启动模式可以在Manifest.xml文件中通过launchMode属性设置,也可以在代码中通过添加Intent的Flag参数来设置
- standard(标准模式)
standard模式每次打开Activity都会新建Activity实例,不管task(任务栈)中有没有该Activity实例,举个栗子:
ActivityA打开ActivityB,然后通过ActivityB打开ActivityA,仍会再创建ActivityA的实例并放入栈中,此时栈的的结构为A->B->A;接着在ActivityA中再打开ActivityA,则栈的结构为:A->B->A->A;如果用户想要回退则会关闭三次ActivityA,体验十分不友好。 - singleTop(栈顶复用模式)
顾名思义如果一个Activity在栈顶,则需要再次打开该Activity时不用新建该Activity实例而是调用该Activity的onNewIntent方法把新的intent传递过来,然后复用该Activity。如果该Activity不在栈顶还是会创建新的Activity实例,下面举例说明:
假设有三个Activity A,B,C并且B的启动模式为singleTop
场景1:任务栈中只有A,此时A启动B。由于B并不在任务栈中,因此会创建B的实例并放入栈顶,任务栈变为A->B效果同standard
场景2:当前任务栈结构为A->B,此时再启动B。由于B已经在任务栈中且位于栈顶,因此会调用B的onNewIntent方法,但不会新建B的实例,任务栈结构仍为A->B
场景3:当前任务栈的结构为A->B->C,此时再启动B。B虽然位于任务栈的中但不在栈顶,此时仍会创建B的实例并放入栈顶,任务栈变为A->B->C->B
singleTop模式下仍会创建多个实例,如果想在同一个任务栈中只有一个实例,就需要使用singleTask模式了 - singleTask(栈内复用模式)
如果一个Activity的启动模式为singleTask,当启动这个Activity时,如果栈中已经有了这个Activity实例,不管该Activity实例在栈中的什么位置都不会新建该Activity的实例,而是调用该Activity的onNewIntent方法并清空该Activity栈上面的其它Activity实例,从而让该Activity位于栈顶。
假设有四个Activity A,B,C,D并且B的启动模式为singleTask
场景1:当前栈中只有A,此时打开B,因为栈中没有B的实例,所以会新建B的实例并放入栈顶,而后栈的结构为A->B
场景2:当前栈的结构为A->B-C-D,此时在D中打开B,由于栈中已有B的实例,所以不会新建B而是调用B的onNewIntent方法,并把B上面的Activity实例清空,而后栈的结构为A->B;
一般应用于App的主页,从任何入口跳转到主页点击返回都能够退出App - singleInstance(单一实例模式)
设置为singleInstance的Activity具有全局唯一性,即在同一时刻安卓系统中仅存在一个该Activity实例,其它应用复用该Activity实例
启动该Activity时如果该Activity实例尚不存在会在新的任务栈中创建该Activity实例,如果该Activity实例已经存在则直接显示该Activity实例
该Activity实例不与其它的Activity实例共存于同一个任务栈中,通过该Activity实例启动的其它Activity,只能在新的任务栈中创建实例。如果已经存在与新启动的Activity的taskAffinity值相同的任务栈,则在该任务栈中创建新启动的Activity实例,否则就在一个新的任务栈中创建实例。
三 LaunchMode与StartActivityForResult
在安卓开发过程中我们常常使用startActivityForResult方法启动Activity并在onActivityResult方法中接收上个页面返回的数据,但你有可能遇到拿不到返回值的情况,可能是因为打开的Activity的启动模式为singleTask。安卓5.0之后安卓的launchMode和onActivityResult发生了一些改变,假如Activity A 启动Activity B:
5.0以前
如果A的launchMode为singleInstance或者B的启动模式为singleTask或者singleInstance则A收不到返回数据;
5.0之后
不管A和B的启动模式是什么,A都可以收到B的返回数据
四 通过设置intent的flag来设置启动模式
系统提供了两种方式来设置启动模式,除了通过在Manifest.xml中设置还可以通过intent的Flag来设置,下面来介绍几种Flag
- FLAG_ACTIVITY_NEW_TASK
这个Flag是我们平时见的最多的,使用一个新的task来启动一个Activity。该flag通常使用在从Service启动Activity的场景,因为在Service中并不存在task,所以该flag用来创建一个新的task并在该task中创建Activity实例。 - FLAG_ACTIVITY_SINGLE_TOP
使用singletop模式启动一个Activity,与指定android:launchMode=“singleTop”效果相同。 - FLAG_ACTIVITY_CLEAR_TOP
使用SingleTask模式来启动一个Activity,与指定android:launchMode=“singleTask”效果相同。
Activity启动模式就介绍完了,在实际开发过程中选择合适的启动模式可以解决一些复杂的问题,例如:某个页面不能重复打开,我们就可以设置其启动模式为singleTask。在明白启动模式的基础上我们可以根据具体需求来选择启动模式。