Activity进阶篇之启动模式
当一个有界面的Android程序运行的时候至少会有Activity,而系统是通过任务栈(一个app程序可以有多个任务栈)来管理这些Activity的,任务栈是一个后进先出的集合。没有销毁的Activity都在任务栈中,新创建的Activity也会添加到任务栈中,而管理规则则是根据启动模式来进行的。
模式介绍
Activity的启动模式有4种:Standard模式、SingleTop模式、SingleTask模式和SingleInstance模式。
Standard模式
系统默认的启动模式,当不设置启动模式的时候,则使用这种启动模式。使用Standard模式时,启动一个Activity就会在任务栈中添加一个Activity实例,不管任务栈中是否有这个Activity实例,都会新创建一个Activity实例放到任务栈中。
SingleTop模式
当使用SingleTop启动模式创建Activity实例时,会先判断任务栈栈顶的Activity是否是将要创建的这个Activity。如果是,则不重新创建了,就使用栈顶的这个Activity实例,同时会调用Activity的onNewIntent(Intent intent)方法,可以在参数intent中获取传递过来的数据(如果设置了);如果不是,则创建一个新的Activity实例,并放到栈顶。需要注意的时,不管任务栈(其他栈和当前栈)是否有该Activity实例,都只判断当前栈的栈顶的是否有该Activity的实例,其他栈和非栈顶的不判断。所以如果其他的任务栈有了该Activity实例,且在栈顶,但是在该任务栈还是会重新创建一个Activity实例放到该栈中。
SingleTask模式
当使用SingleTask模式的时候,任务栈中有且只有一个该Activity的实例,当设置成这种模式的时候,有好几种情况:
1.如果任务栈中没有该Activity实例,则跟Standard模式一样;
2.如果任务栈栈顶有了该Activity实例,则跟SingleTop模式一样;
3.如果当前栈有该Activity的实例,且不在栈顶,那么会把处于该Activity实例上面的所有Activity实例都清空掉,即销毁掉,然后调用该Activity实例的onNewIntent(Intent intent)方法。
4.如果当前栈没有该Activity实例,但是在别的任务栈中有该Activity的实例,并且处于栈顶,那是怎样一种情况呢?当一个Activity启动另一个Activity且要启动的Activity实例在另外一个任务栈中已存在且处于栈顶时,刚开始当前任务栈是处于前台任务栈的(非前台任务栈不符合场景),另外一个任务栈处于后台任务栈,启动后,后台任务栈会被调用到前台变成前台任务栈,且所有Activity的顺序是不变的,按返回键还是一个一个返回。举个例子:前台任务栈有A和B,B处于栈顶,后台任务栈有C和D,C处于栈顶,当B启动C后,Activity的顺序是ABDC,按返回键是先退回到D,再按返回键退回到B,然后是A,是这么一个顺序,但是任务栈还是没有变的。
5.如果当前栈没有该Activity实例,但是在别的任务栈中有该Activity实例,且不处于栈顶,这个时候会跟上面一样,唯一不同的就是会把处于后台栈中该Activity实例上方的所有Activity实例推出去。
6.如果将要启动的Activity实例所在的任务栈都没有,那么就先创建一个任务栈,然后根据是否有该Activity实例的情况来把该Activity实例放到该任务栈中。
SingleInstance模式
SingleInstance模式是特殊的SingleTask模式,它会重新创建一个任务栈,且该任务栈中只有它一个Activity,同时也不能往这个任务栈中添加Activity实例。
使用场景
正常情况下一般都使用Standard模式;一般点击通知栏打开Activity需要使用SingleTop模式,因为这个时候Activity已经处于栈顶了,当又来一个消息通知时,不需要再打开一个Activity,只需要更新界面就可以了;主页一般使用SingleTask模式,当把主页退出时,会把它上面的所有Activity实例都退出,相当于退出程序了;启动页或者广告页以及聊天界面使用SingleInstance,因为这个在程序中只能有且只有一个界面,有两个以上的话就感觉不对了。启动模式也有其他的一些使用场景,根据需求可以知道具体使用哪种启动模式
设置方式
有两种方式可以设置启动方式:一、在AndroidManifest文件的activity节点设置launchMode属性即可设置启动方式;二、启动Activity时,intent的addFlags方法或者setFlags方法可以设置启动方式。第二种设置方式优先于第一种设置方式,但是第二种方式不能够设置SingleInstance模式,第一种不能设置FLAG_ACTIVITY_CLEAR_TOP模式