Activity的启动模式也是一个难点,主要原因是形形色色的启动模式和标志太容易被混淆了。
Activity的LaunchMode
我们知道,默认情况下,当我们多次启动同一个Activity的时候,系统会创建多个实例并把他们一一放入任务栈中,当我们点击Back键时,会发现这些Activity会一一回退。任务栈是一种“后进先出”的栈结构。没按一次会有一个Activity出栈,知到栈空为止,当栈中午任何Activity的时候,系统就会自动回收这个任务栈。默认模式在多次启动同一个Activity时会创建多个实例,这样会显得很傻,当然为了避免这种情况,Android在设计的时候提供了启动模式来修改系统默认行为。目前有四种启动模式:standard、singleTop、singleTask、singleInstance。
1,standard:标准模式
也就是系统的默认模式,每启动一个Activity都会创建新的实例,不管实例是否已经存在。这是一种典型的多实例多实现,一个任务栈中可以有多个实例,每个实例也可以属于不用的任务栈。在这种模式下,谁启动这个Activity,那么这个Activity就运行在启动它的那个Activity所在的任务栈中,比如Activity A启动Activity B(B是标准模式) ,那么B就会进入A所在的栈中。如果当我们用ApplicationContext去启动standard模式的Activity的时候会报错。这是因为standard的Activity默认进入启动它的Activity所属的任务栈中,但是由于非Activity的Context并没有任务栈,所以会报错。
2,singleTop:栈顶复用模式
在这种模式下,如果新的Activity已经位于任务栈的栈顶,那么此Activity不会被重新创建,同时它的onNewIntent 方法会被回调,通过此方法的参数我们可以取出当前请求的信息。需要注意的是,这个Activity的onCreate和onStart不会被系统调用,因为它并没有发生变化,如果新的Activity的实例已经存在但是不是位于栈顶,那么新Activity还是会重建。比如,假设栈内是ABCD,其中ABCD为四个Activity,A在栈顶,D在栈顶,这时候再启动D,如果D启动模式为singleTop则栈内的情况还是ABCD。
3,singleTask:栈内复用模式
这是一种单实例模式,在这种模式下,只要Activity在一个栈中存在,那么多次启动这个Activity都不会重建,和singleTop一样,系统会调用onNewIntent。具体一点,当一个具有singleTask模式的Activity请求启动以后,比如ActivityA,系统会首先寻找是否存在ActivityA想存在的任务栈,,如果不存在,那就重新创建一个任务栈,然后创建A的实例后把A放入栈中。如果存在A所需要的任务栈,这时候要看A是否在任务栈中有实例存在 。如果存在,那么系统会吧A调到栈顶并调用它的onNewIntent方法,如果实例不存在,就创建A的实例并把A压入栈中。
举几个例子:
(1)比如现在任务栈S1中的情况为ABC,这时候Activity D以singleTask模式请求启动,其所需要的任务栈为S2,由于S2和D的实例均不存在,所以系统会先创建任务栈S2,然后创新D的实例并将其入栈到S2中。
(2)另一种情况,假设D所需的任务栈为S1,其他情况如1所示,因为S1已经存在了,所以系统会直接创建D的实例并将其入栈到S1中。
(3)如果D所需的任务栈为S1,并且当前S1栈内情况为ADBC,根据栈内复用的原则,此时D不会重新创建,系统会把D切换到栈顶,并调用其onNewIntent方法。同时singleTask默认具有clearTop的效果,会导致D上面的Activity全部出栈,于是最终S1栈内的情况为AD。
4,singleInstance:单实例模式
这是一种加强版的singleTask模式,它除了具有singleTask模式的所有特点外,还加强了一点,那就是具有这种模式的Activity只能单独位于一个任务栈中,换句话说比如ActivityA是singleInstance 模式,当A启动后,系统会为它创建一个新的任务栈,然后A独自在这个任务栈中,由于栈内复用的特性,后续的请求均不会再创建新的Activity,除非这个独特任务栈被系统销毁了。
在singleTask启动模式中,多次提到了某个Activity所需的任务栈,其实也就是TaskAffinity,这个参数标识着一个Activity所需的任务栈的名称。默认情况下,所有Activity所需的任务栈名称都是应用的包名,当然我们也可以为Activity单独指定TaskAffinity属性,这个属性不能与包名相同,否则等于没有指定。TaskAffinity属性主要和singleTask启动模式或者allowTaskReparenting 属性配对使用,在其他情况下使用无意义。另外任务栈分为前台任务栈和后台任务栈。后台任务栈中的Activity位于暂停状态,用户可以通过切换将后台任务栈再次调到前台。
当TaskAffinity和singleTask启动模式配对使用的时候,他是具有该模式的Activity的目前任务栈的名字,待启动的Activity会运行在名字和TaskAffinity相同的任务栈中。
如何给Activity指定启动模式呢?有两种方法。
第一种在AndroidManifest中为Activity指定启动模式
<activity
android:launchMode="singleTask"
android:name="com.demo.main.AddDeviceActivity" >
第二种是通过Intent中设置标签位来为Activity指定启动模式
Intent intent=new Intent();
intent.setClass(MainActivity.this,AddDeviceActivity.class);
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
startActivity(intent);
Activity中的Flags
1,FLAG_ACTIVITY_NEW_TASK
这个标记位的作用是为Activity指定“singleTask”启动模式,效果与在XML中指定该启动模式相同。
2,FLAG_ACTIVITY_SINGLE_TOP
这个标记位的作用是为Activity指定“singleTop”启动模式,效果与在XML中指定该启动模式相同。
3,FLAG_ACTIVITY_CLEAR_TOP
具有此标记的Activity启动时,处于同一任务栈中所有位于它之上的Activity都要出栈。这个模式一般需要和FLAG_ACTIVITY_NEW_TASK配合使用,在这种情况下,如果被启动的Activity已经存在,那么系统就会调用它的onNewIntent方法。如果采用standard模式启动,那么它连同它之上的Activity都要出栈,系统会创建新的Activity实例放入栈顶。
4,FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS
具有这个标记的Activity不会出现在历史Activity的列表中,不希望用户返回该Activity时可以用这个标记。等同于在XML中设置Activity的属性android:autoRemoveFromRecents="true"。