Activity 的4种启动模式(launchMode),像这种姿势,基本上是面试必问的,在这里把它给彻底搞懂,搞透!搞怀孕,呃,不对。
官方说法
先给出 Google 的文档,有兴趣的可以先去看看。任务和返回栈 这个是翻译版,虽然是官方翻译的,但是有些地方看起来还是怪怪的,看不惯的同学可以科学上网去看原版。
这里有一句话很关键:任务是指在执行特定作业时与用户交互的一系列 Activity。 这些 Activity 按照各自的打开顺序排列在堆栈(即返回栈)中。 这句话讲得很模糊,不知道任务和返回栈是一个概念。在我看来,就把 task 理解成一个堆栈,是用来存放 Activity 的就可以了。如果 task 中的 Activity 都销毁了,task 也不存在了。但是只要 task 中还有一个 Activity 存在,task 就还存在。所以也不能说 task 是要多个 Activity 才能存在。
task 是怎么工作的
上面也说了,task 是以堆栈的方式来管理 Activity 的。当应用启动的时候,会创建一个 task,然后把主 Activity 放进 task 中。继续启动 BCD,现在 task 中有4个 Activity: Main、B、C、D,他们以各自打开的顺序放在 task 中。
现在按返回键,D 出栈,taks 中还有:Main、B、C。再按3次返回栈,所有Activity 都销毁出栈,然后 task 也不存在了,下次启动的时候会再创建一个新的 task。是不是感觉很熟悉呢,没错,我们平时就是这么使用 APP 的,炒鸡简单是不是。
来一张偷来的图片,是不是看起来更形象了呢
好了,现在知道 task 是什么东西了,接下来就轮到4个启动模式了,这4个启动模式都跟 task 有紧密关系。
4个启动模式
其实,应该先想想,为什么会有4个启动模式。那肯定是因为默认的启动模式不能满足我们的需求,比如你想启动一个 Activity 的时候,想把它放到一个新的 task 中。又或者当 task 中已经有这个 Activity 了,你想直接使用这个 Activity 就行了,而不是在启动一个新的。这些都可以通过配置 Activity 的启动模式实现。
配置 Activity 的启动模式有2种方式,一种是在 manifest 的 <activity>
标签中配置,另外一种是使用 intent 标志。并且 使用intent 标志定义优先级会比在 manifest 中定义的要高。就是说会覆盖的意思。
1. standard(默认模式)
默认模式,Activity 在 task 中可以有多个实例。例如:现在有一个 task 和一个启动模式是 standard 的 Activity A,那么在task中可以有多个 A 的实例。这个很简单是不是
2. singleTop
如果当前 task 的顶部已存在 Activity 的一个实例,则系统会通过调用该实例的 onNewIntent() 方法向其传送 Intent,而不是创建 Activity 的新实例。 记住,这个是要在顶部的时候,才不会创建新实例,如果不是在顶部,那么它还是会创建新的 Activity。
例如:当前 task 中有 A B C 三个 Activity,如果 C 的启动模式是 singleTop 那么当调用 startActivity 启动 C 的时候,不会创建新的 Activity,而是通过 onNewIntent()方法 把这个 intent 传递进去。 taks 中还是 A B C 三个 Activity。 这个也很简单是不是。
上图所示: B 不是在顶部,所以还是会创建新的实例
3. singleTask
这个就厉害了,一个Activity 声明为 singleTask,那么它在一个 task 中,就只能有一个实例。其实从名字就可以看出来了,single in task 。并且它还会把在它之上的 Activity 全都踢出栈,让它自己升上顶部,霸道总裁既视感啊。例如:task 中有三个 Activity ABC,A的启动模式是 singleTask。
然后我们调用 startActivity(A) 启动 A,这时系统会调用 A 的 onNewIntent() 方法向其传送 Intent。把 B C 出栈,A上到栈顶。
4. singleInstance
这个比 singleTask 还要屌一点,它也是在一个 task 中只能有一个实例,并且 singleInstance 会独自占用一个 task,不与其他任何 Activity 共享。孤独终老的赶脚啊!