Activity的启动模式
<关于Activity的LaaunchMode>
因为一个应用里多次启动同一个Activity会重复创建,所以Andorid提供了四种启动模式
standard:标准模式
每次启动一个Activity都会重新创建,不管这个Activity的实例是否已经存在。创建的Activity生命周期和常规的一样。这是一种典型的多实例实现,一个任务栈可以有多个实例,每个实例也可以属于不同的任务栈,比如A启动了B,那么这个B的实例是属于A的任务栈, C也可以启动B,那么这个B的实例是属于C所在的任务栈的。
这里注意,非Activity的Context(如ApplictationContext)是没有任务栈的,所以用之去启动standard模式会报错,解决方法使为待启动的Activity指定FLAG_ACTIVITY_NEW_TASK,为其创建一个任务栈。singleTop:栈顶复用模式
这种模式下如果Activity已经位于栈顶则这个Activity不会重新创建,同时它的onNewIntent方法会被调用,通过此方法的参数我们可以取得当前的请求信息。如果新的Activity不是位于栈顶,那么它就会重新被创建。如果目前的栈为ABCD,模式为standard,要创建一个D,则栈变为ABCDD,如果为singleTop则为ABCD。singleTask:栈内复用模式
这是一种单实例模式。Activity启动时要先查找栈里面是否有实例,如果有则将其调为栈顶,不创建新的实例,如果不存在则创建新的实例压入栈中。
①S1栈中Activity为ABC,想启动D,则创建实例压栈,S1变为ABCD。
②S1栈中为ABC,但D想要的栈为S2,S2不存在,所以S2和D同时创建并将D压入S2栈中。
③S1栈中为ABCD,想在S1中启动B,则将CD出栈,S1栈中变成AB。
(以上均在singTask模式下进行)singleInstance:单实例模式。
是一种加强的singleTask模式,所以它具有singleTask模式的功能,还加强了一点,就是此种模式的Activity只能单独的位于一个任务栈。Activity A这种模式下只能单独的位于栈S1中,Activity B只能单独的位于栈S2中。
<关于TaskAffinity>
它标志了一个Activity所需要任务栈的名字。默认情况下都是Activity应用的包名。所以它和singleTask配对使用。还有一个参数allowTaskReparenting,如果其为True并且singleTask下,那么 当一个Activity A启动B中的一个Activity,当B启动时,这个Activity的任务栈会从A转移至B。
可以通过从Menifest中指定启动模式,也可以从Acitivity的addFlags中设置标志位来指定启动模式。从优先级来说,后者要高于前者。并且两种方式的限定范围也有所不同,前者不可以指定FLAG_ACTIVITY_CLEAR_TOP标识,后者也不能为Activity指定singleInstance模式。
书中为singleTask介绍了一个例子来说明其运作方式:
A为standard模式,所以其TaskAffinity为默认包名S1,B和C均为singleTask模式,TaskAffinity为S2。A先启动B,B然后启动C,这个时候栈的情况是:S1中只有A,因为B和C需要S2任务栈,所以B创建了S2栈,S2栈中为BC。 然后C启动A,这个时候因为A是standard模式,所以它会进入到创建它的那个Actvitiy即C的栈中,所以S2栈变为BCA,S1栈为A。接着A启动B,因为B为singleTask模式,所以S2中的CA均弹出栈让B到栈顶,此时S2栈为B。接着按下返回键,S2栈的Actviity被取完,后台的S1栈被放入前台。而S1只有A,但这个A并不是S2栈而是S1栈的。接着A中按下返回键,所有栈的Activity被取完,返回桌面。
这里给个别的例子的singleTask的任务栈图理解一下:
<关于Activity的Flags>
- FLAG_ACTIVITY_NEW_TASK
指定为singleTask模式 - FLAG_ACTIVITY_SINGLE_TOP
指定为singleTop模式 - FLAG_ACTIVITY_CLEAR_TOP
和FLAG_ACTIVITY_NEW_TASK配合使用,在同一个栈位于它上面的Activity都要出栈。如果被启动的Activity为standard模式,则它连同它之上的Activity都出栈,然后重新创建压栈。singleTask自动模式都默认有这个标志位的效果 - FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS
具有这个标志的Activity不会出现在历史Activity列表中。