Android进阶(二)Activity的启动模式
在上篇文章中我们介绍了Activity的生命周期,那么接下来就要介绍同样是重点而且是难点的Activity的启动模式!
说到启动模式,对于很多初学者来说,可能仅仅只是听说过这种概念而已,并没有亲自验证过每个启动模式有什么不同!在详细介绍每个启动模式之前,我们有必要了解一下启动模式为什么会出现,它的意义是什么?我们都知道,在正常情况下,每启动一个Activity,系统都会创建一个实例,并把实例放到任务栈中,那么当我们多次启动同一个Activity时,系统依然这么做,是不是就不合理了,我们都能想到的问题,Google的大牛们咋么会想不到呢,于是乎就有了启动模式的出现,通过启动模式我们可以去修改系统的这种行为。
启动模式有四种,分别是:standard,singleTop,singleTask和singleInstance,接下来会分别详细的介绍每种启动模式的含义以及行为。
standard:标准模式
标准模式,顾名思义,就是在默认情况下Activity的启动模式,也就是我们上文所提到的每次启动一个Activity,都会重新创建一个新的实例,并把该实例添加到任务栈中。被创建的Activity的生命周期符合典型情况下Activity的生命周期。在标准模式下,谁启动了Activity,那么这个Activity就运行在启动它的那个Activity所在的栈中。举个简单的例子,假如Activity A启动了ActivityB,那么B就会和A在相同的栈中。注意:当我们用ApplicationContext去启动standard模式的Activity的时候会报错,错误如下:AndroidRuntime:android.util.AndroidRuntimeException:Calling startActivity from outside of an Activity context requires the FLAG_ACTIVITY_NEW_TASK flag. Is this really what you want?
这个错误应该好多初学者都会碰到,特别是不按套路出牌的初学者(这样就对了),这是因为标准模式的Activity默认会进入启动它的Activity所在的任务栈,但是由于非Activity类型的Context(如ApplicationContext)并没有所谓的任务栈,所以就会有这个问题了,那么怎么解决呢,其实也很简单,新建一个任务栈就可以了,怎么新建一个任务栈呢,需要添加一个标记位FLAG_ACTIVITY_NEW_TASK,添加完标记位后,启动Activity的时候就会为它创建一个新的任务栈。但是此时Activity实际上是以singleTask模式启动。
singleTop:栈顶复用模式
这种模式,就是说如果启动一个Activity,该Activity正好又位于栈顶,这时此Activity就不会被重建,同时它的onNewIntent(Intent intent)方法将会被调用,通过该方法,就可以获取当前请求的信息(通过intent传递过来的数据)。而此时该Activity的onCreate()方法和onStart()方法是不会执行的。但是如果被启动的Activity不位于栈顶,那么该Activiy还是会被重建。
singleTask:栈内复用模式
这是一种单实例模式,在这种模式下,只要Activity在一个栈(任何栈一个)中存在,那么多次启动该Activity都不会创建新的实例,并且onNewIntent(Intent intent)方法会被调用。举例,假如ActivityA是singleTask模式,那么当ActivityA的请求启动后,系统首先会寻找是否存在A想要的任务栈,如果不存在,就重新创建一个任务栈,然后创建A的实例把A放到栈中。如果存在A所需的任务栈,就要看栈中是否有A的实例,如果实例存在,系统会把A调到栈顶并调用它的onNewIntent()方法,如果过实例不存在,就创建A的实例,并把A压入栈中。
singleInstance:单实例模式
该模式是在singleTask模式上进行了进一步的加强,具有此模式的Activity只能单独的位于一个任务栈中。举个例子,比如ActivityA是singleInstance模式,当A启动后,系统会为它创建一个新的任务栈,然后A独自在这个新的任务栈中,由于栈内复用的特性,后续的请求均不会创建新的Activity,除非这个独特的任务栈被系统销毁了。
介绍完启动模式之后,接下来就说说如何在项目中使用启动模式。给Activity设置启动模式主要有两种方式,第一种在AndroidMenifest文件中设置。如下图所示。
第二种方式是通过设置标志位来为Activity指定启动模式:
第二种方式这两种方式都可以为Activity指定启动模式,但是二者还是有区别的。优先级上第二种方式(即通过标记位)的方式要高于第一种方式,当两种方式同时存在时,以第二种方式为准。