众所周知,在android系统中是通过任务栈来管理我们的Activity的。Task是抽象的概念,指带了一组Activity,它们为实现用户的某个操作目的而聚在了一起,可以来自不同应用。
Android系统下,当用户为了完成某一个功能可能需要进行多个Actvitiy间的跳转才能达到目的,这些Activity的跳转序列就被Android抽象成了一个Task。而这一组Actvitiy实例都被放到了同一个栈中,先启动的Activity位于栈底,最后到达的Activity位于栈顶。一般而言Task的启动点都是从home 界面算起,点击Launcher界面的应用icon,如果之前没有启动过,则系统新建一个Task,刚启动应用的主Activiy被压入栈底,栈内的Activity是不会在内部重新排列的,只能按先入后出的顺序呈现,当用户连续按back键,使得栈底的Activity也pop出栈,栈为空了,这时该Task结束。但是这样的话,如果我们多次启动同一个Activity,就会重复的去创建该Activity的实例,由此Android提出了4种启动模式来优化这个问题。
究竟是哪4种呢? 有什么区别?
进入正题,分别是standard(默认启动模式),singleTop(栈顶单例模式),singleTask(栈内复用模式),singleInstance(全局单例模式)。
接下来我们分别就这4种模式进行区分?
standard(默认模式):
此模式是默认的启动模式,即不显示的进行说明系统会默认使用此加载模式。用这种模式加载Activity,每次启动该Activity都会创建该Activity的实例无论栈中是否已经存在该Activity实例,这种模式下,谁启动了该模式的Activity,该Activity就属于启动它的Activity的任务栈中。这个Activity它的onCreate(),onStart(),onResume()方法都会被调用。
AndroidManifest.XML配置如下:
这种模式比较简单,不用多说。
singleTop(栈顶复用模式):
简单来说,如果我要进入的Activity如果已经位于栈顶,则直接复用不会创建新的实例子,如果栈中不存在该Activity的实例,则就如standard启动模式。可是别忘了还有一宗情况,如果该Activity不在栈顶,但是栈中存在该Acticity的实例子呢?比如我从MainActivity进入到ActivityA,我再从ActivityA进入到B,此时栈中从栈底到栈顶的顺序为MainActivity->A->B,此时若从B中再次进入A,系统会重新创建A的实例,并移到栈顶。需要注意的是,这个ActivityA它的onCreate(),onStart()方法不会被调用。
总结一下就是:standard启动模式是默认的启动模式,每次启动一个Activity都会新建一个实例不管栈中是否已有该Activity的实例。
而singleTop模式分3种情况
当前栈中已有该Activity的实例并且该实例位于栈顶时,不会新建实例,而是复用栈顶的实例,回调onNewIntent方法
当前栈中已有该Activity的实例但是该实例不在栈顶时,其行为和standard启动模式一样,依然会创建一个新的实例
当前栈中不存在该Activity的实例时,其行为同standard启动模式
但是这两种模式都是基于在同一个任务栈中进行操作,不会创建新的栈。
AndroidManifest.XML配置如下:
singleTask(栈内复用模式):
顾名思义:这种启动模式,只要启动的Activity位于栈内,就不会创建新的实例子而是复用,但是需要注意的一点是,如果该Activity不是位于栈顶,则首先会将该Activity上所有的Activity出栈,然后才复用并回调该实例的onNewIntent()方法。
AndroidManifest.XML配置如下:
singleInstance(全局单例模式):
该模式具备singleTask模式的所有特性外,与它的区别就是,这种模式下的Activity会单独占用一个Task栈,具有全局唯一性,即整个系统中就这么一个实例,由于栈内复用的特性,后续的请求均不会创建新的Activity实例,除非这个特殊的任务栈被销毁了。以singleInstance模式启动的Activity在整个系统中是单例的,如果在启动这样的Activiyt时,已经存在了一个实例,那么会把它所在的任务调度到前台,重用这个实例。
简而言之,采用这种启动模式启动一个Activity,会创建一个新的栈存放该Activity的实例,并且这个栈中有且仅有这一个实例。
AndroidManifest.XML配置如下: