• 8.3 AndroidMainifest启动模式" level="2">8.3 AndroidMainifest启动模式
    • 8.3.1 standard" level="3">8.3.1 standard
    • 8.3.2 singleTop" level="3">8.3.2 singleTop
    • 8.3.3 singleTask" level="3">8.3.3 singleTask
    • 8.3.4 singleInstance" level="3">8.3.4 singleInstance

    8.3 AndroidMainifest启动模式" class="reference-link">8.3 AndroidMainifest启动模式

    下面来看看这些“特权”都有哪些能力。Android开发者在AndroidMainifest文件中一共设计了四种启动模式,如下所示。

    • standard
    • singleTop
    • singleTask
    • singleInstance

    这四种启动模式都具有不同的功能,下面我们一一学习一下。

    8.3.1 standard" class="reference-link">8.3.1 standard

    默认的启动模式,如果不指定Activity的启动模式,则使用这种方式启动Activity。这种启动模式每次都会创建新的实例,每次点击standard模式创建Activity后,都会创建新的MainActivity覆盖在原Activity上,结构如图8.7所示。

    8.3 AndroidMainifest启动模式 - 图1 图8.7 standard启动模式

    8.3.2 singleTop" class="reference-link">8.3.2 singleTop

    如果指定启动Activity为singleTop模式,那么在启动时,系统会判断当前栈顶Activity是不是要启动的Activity,如果是则不创建新的Activity而直接引用这个Activity;如果不是则创建新的Activity。这种启动模式通常适用于接收到消息后显示的界面,例如QQ接收到消息后弹出Activity,如果一次来10条消息,总不能一次弹10个Activity,这种启动模式的任务栈结构如图8.8所示。

    8.3 AndroidMainifest启动模式 - 图2 图8.8 singleTop启动模式

    这种启动模式虽然不会创建新的实例,但是系统仍然会在Activity启动时调用onNewIntent()方法。举个例子来说,如果当前任务栈中有A、B、C三个Activity,而且C的启动模式是singleTop的,那么这时候如果再次启动C,那么系统就不会创建新的C的实例,而是会调用C的onNewIntent()方法,当前任务栈中依然是A、B、C三个Activity。

    8.3.3 singleTask" class="reference-link">8.3.3 singleTask

    singleTask模式与singleTop模式类似,只不过singleTop是检测栈顶元素是否是需要启动的Activity,而singleTask是检测整个Activity栈中是否存在当前需要启动的Activity。如果存在,则将该Activity置于栈顶,并将该Activity以上的Activity都销毁。不过这里是指在同一个App中启动这个singleTask的Activity,如果是其他程序以singleTask模式来启动这个Activity,那么它将创建一个新的任务栈。不过这里有一点需要注意的是,如果启动的模式为singleTask的Activity已经在后台一个任务栈中了,那么启动后,后台的这个任务栈将一起被切换到前台,借助官网上的一张图可以更好地理解这一过程,如图8.9所示。

    当Activity2启动ActivityY(启动模式为singleTask)时,它所在的Task都被切换到前台,且按返回键返回时,也会先返回ActivityY所在Task的Activity,这一点比较难以理解,希望大家能通过图8.9认真理解这一过程。

    8.3 AndroidMainifest启动模式 - 图3 图8.9 singleTask启动模式特例

    可以发现,使用这个模式创建的Activity不是在新的任务栈中被打开,就是将已打开的Activity切换到前台,所以这种启动模式通常可以用来退出整个应用:将主Activity设为singleTask模式,然后在要退出的Activity中转到主Activity,从而将主Activity之上的Activity都清除,然后重写主Activity的onNewIntent()方法,在方法中加上一句finish(),将最后一个Activity结束掉。

    8.3.4 singleInstance" class="reference-link">8.3.4 singleInstance

    singleInstance这种启动模式和使用的浏览器工作原理类似。在多个程序中访问浏览器时,如果当前浏览器没有打开,则打开浏览器,否则会在当前打开的浏览器中访问。申明为singleInstance的Activity会出现在一个新的任务栈中,而且该任务栈中只存在这一个Activity。举个例子来说,如果应用A的任务栈中创建了MainActivity实例,且启动模式为singleInstance,如果应用B也要激活MainActivity,则不需要创建,两个应用共享该Activity实例。这种启动模式常用于需要与程序分离的界面,如在SetupWizard中调用紧急呼叫,就是使用这种启动模式。

    关于singleTop和singleInstance这两种启动模式还有一点需要特殊说明:如果在一个singleTop或者singleInstance的ActivityA中通过startActivityForResult()方法来启动另一个ActivityB,那么系统将直接返回Activity.RESULT_CANCELED而不会再去等待返回。这是由于系统在Framework层做了对这两种启动模式的限制,因为Android开发者认为,不同Task之间,默认是不能传递数据的,如果一定要传递,那就只能通过Intent来绑定数据。