启动模式简单地说就是Activity启动时的策略,可以通过Android Manifest.xml中的Activity标签的android:launchMode属性进行设置;启动模式有4种,分别为standard、singleTop、singleTask、singleInstance;
standard:模式启动模式,每次激活Activity时都会创建Activity,并放入任务栈中。
singleTop:如果在任务的栈顶正好存在该Activity的实例,就重用该实例,否者就会创建新的实例并放入栈顶(即使栈中已经存在该Activity实例,只要不在栈顶,都会创建实例)。
singleTask:如果在栈中已经有该Activity的实例,就重用该实例(会调用实例的onNewIntent())。重用时,会让该实例回到栈顶,因此在它上面的实例将会被移除栈。如果栈中不存在该实例,将会创建新的实例放入栈中。
singleInstance:在一个新栈中创建该Activity实例,并让多个应用共享该栈中的该Activity实例。一旦该模式的Activity的实例存在于某个栈中,任何应用再激活改Activity时都会重用该栈中的实例,其效果相当于多个应用程序共享一个应用,不管谁激活该Activity都会进入同一个应用中。
"standard" 和"singleTop"可以存在多个实例。
"singleTask" 和 "singleInstance"在一个设备上的任意时刻只能存在一个实例,singleTask允许其他的Activity和自己共用一个Task。而“singleInstance”模式的Activity只能自己拥有这个Task。
设置了"singleTask"启动模式Activity,也并不是总是在Task的顶部,而是在启动后需要确保它在Task的顶部。
关于singleTask比较难于理解,主要原则如下:
1. 设置了"singleTask"启动模式的Activity,它在启动的时候,会先在系统中查找属性值affinity等于它的属性值taskAffinity的任务存在;如果存在这样的任务,它就会在这个任务中启动,否则就会在新任务中启动。因此,如果我们想要设置了"singleTask"启动模式的Activity在新的任务中启动,就要为它设置一个独立的taskAffinity属性值。taskAffinity属性值默认为父标签application的taskAffinity属性值,如果标签application的taskAffinity也没有配置,于是它们就默认为包名。
2. 如果设置了"singleTask"启动模式的Activity不是在新的任务中启动时,它会在已有的任务中查看是否已经存在相应的Activity实例,如果存在,就会把位于这个Activity实例上面的Activity全部结束掉,即最终这个Activity实例会位于任务的堆栈顶端中。
另外,Intent的标志设定为Intent. FLAG_ACTIVITY_NEW_TASK并启动Activity时并不意味着一定会启动一个新的Task,而是首先会查找是否存在和被启动的Activity具有相同的亲和性的任务栈(即taskAffinity),如果有,刚直接把这个栈整体移动到前台,并保持栈中的状态不变,即栈中的Activity顺序不变,并把被启动的Activity压入这个Task栈中,如果没有,则新建一个Task来存放被启动的Activity。
具体分析时,可以通过如下命令的输出信息中查找ActivityStack中的TaskRecord
adb shell dumpsys activity