Android启动模式

说明一下, 大家看这个帖子之前对Task,有一定的了解。而且看完这个之后,最好再看看taskAffinity属性,因为在实际应用中他们一般是联合使用的。
在Android的中Activity 有4种启动方式 :"standard"、"singleTop"、"singleTask"、"singleInstance"。
下面就分轻重点讲讲这些启动方式。
1、standard启动方式
这种方式也是系统默认的。说的简单点,我们在使用startActivity()方法的时候,系统就会启动 一个新的Activity实例,不管它先前启动没有。如果想知道具体细节可以查看这个帖子 http://marshal.easymorse.com/archives/2950,我在这里就不细细讲解了。
配图说明:

Android启动模式_第1张图片

2、singleTop启动方式
如果大家把standard启动方式看懂了,这个我就说一下区别,大家就明白了。如果我们要调用的 Activity在Task栈顶,系统不会像在standard方式下创建一个新的实例,而是调用 onNewIntent()方法,来引导intent到栈顶的实例。如果调用的 Activity不在Task栈顶 ,系统此时调用方式与standard方式下的一模一样。
举个例子:

现在task栈元素为A-B-C-D(D在栈顶),这时候给D发一个启动intent,如果D是 “standard”的,则生成D的一个新实例,栈变为A-B-C-D-D。

如果D是singleTop的话,则不会生产D的新实例,栈状态仍为A-B-C-D

如果这时候给B发Intent的话,不管B的launchmode是”standard” 还是 “singleTop” ,都会生成B的新实例,栈状态变为A-B-C-D-B。

注意:在一个Activity处理新的intent的时候,用户不能使用Back button来返回到先前的Activity。直到新的intent到达onNewIntent()方法中后,才可以。

3、singleTask启动方式

如果已经有其他的Task并且包含Activity实例,那就直接调用那个Task的实例。(只有一个Task中会有) 而不是创建一个新的实例。重点:假如你启动一个singleTask的Activity,同时这个Activity在background task中,那么整个background task都会被调入到前台来。下面这幅图就很好的说明这个问题。

有人问如果我调用图中的Activity X,会怎样?系统会先把Activity Y销毁,然后再将Activity X放到前台。

4、singleInstance启动方式

singleInstance 新建一个Task,且在该Task中只有它的唯一一个实例。 (只有一个Task会有,且该Task中只有它)。“singleInstance”是其所在栈的唯一activity,它会每次都被重用。

总结:

standard  每次都会新建,每个Task都可以有,且每个Task都可以有多个实例(每个Task都可以有,且可以有多个)
singleTop 当前实例如果在栈顶,就不新建实例,调用其OnNewIntent。 如不在栈顶,则新建实例 (每个Task都可以有,且可以有多个,在栈顶时可复用)
singleTask 新建一个Task,如果已经有其他的Task并且包含该实例,那就直接调用那个Task的实例。(只有一个Task中会有)
singleInstance 新建一个Task,且在该Task中只有它的唯一一个实例。 (只有一个Task会有,且该Task中只有它)

intent的flag可以覆盖activity的启动模式:

FLAG_ACTIVITY_CLEAR_TOP

   例如现在的栈情况为:A B C D D此时通过intent跳转到B,如果这个intent添加FLAG_ACTIVITY_CLEAR_TOP标记,则栈情况变为:A B。如果没有添加这个标记,则栈情况将会变成:A B C D B。也就是说,如果添加了FLAG_ACTIVITY_CLEAR_TOP标记,并且目标Activity在栈中已经存在,则将会把位于该目标activity之上的activity从栈中弹出销毁。这跟上面把BLaunch mode设置成singleTask类似。

 

FLAG_ACTIVITY_NEW_TASK

   例如现在栈1的情况是:A B CC通过intent跳转到D,并且这个intent添加了FLAG_ACTIVITY_NEW_TASK标记,如果D这个ActivityManifest.xml中的声明中添加了Task affinity,并且和栈1affinity不同,系统首先会查找有没有和DTask affinity相同的task栈存在,如果有存在,将D压入那个栈,如果不存在则会新建一个Daffinity的栈将其压入。如果DTask affinity默认没有设置,或者和栈1affinity相同,则会把其压入栈1,变成:A B C D,这样就和不加FLAG_ACTIVITY_NEW_TASK标记效果是一样的了。注意如果试图从非activity的非正常途径启动一个activity,比如从一个service中启动一个activity,则intent必须要添加FLAG_ACTIVITY_NEW_TASK标记。

 

FLAG_ACTIVITY_NO_HISTORY

   例如现在栈情况为:A B CC通过intent跳转到D,这个intent添加FLAG_ACTIVITY_NO_HISTORY标志,则此时界面显示D的内容,但是它并不会压入栈中。如果按返回键,返回到C,栈的情况还是:A B C。如果此时D中又跳转到E,栈的情况变为:A B C E,此时按返回键会回到C,因为D根本就没有被压入栈中。

 FLAG_ACTIVITY_SINGLE_TOP:和上面ActivityLaunch modesingleTop类似。如果某个intent添加了这个标志,并且这个intent的目标activity就是栈顶的activity,那么将不会新建一个实例压入栈中。

 /////////////////////////////////////////////////////////////////////////////////////////////////Activity的主要属性:(在 AndroidManifest.xml<activity  android:XXX  />

allowTaskReparenting:设置成true时,和IntentFLAG_ACTIVITY_NEW_TASK标记类似。

alwaysRetainTaskStat:   如果用户长时间将某个task移入后台,则系统会将该task的栈内容弹出只剩下栈底的activity,此时用户再返回,则只能看到根activity了。如果栈底的activity的这个属性设置成true,则将阻止这一行为,从而保留所有的栈内容。

clearTaskOnLaunch:根activity的这个属性设置成true时,和上面的alwaysRetainTaskStat的属性为true情况搞好相反。

finishOnTaskLaunch:对于任何activity,如果它的这个属性设置成true,则当task被放置到后台,然后重新启动后,该activity将不存在了

补充说明,里面遗漏的东西。
用户一定要能够离开一个task,和再回到这个task。所以,"singleTask"和"singleInstance",只能用作启动Activity的属性。如果不这样,我们在启动一个singleTask的Activity之后,使用Home键到主界面,就会回不去了,因为它不代表一个应用的启动。

原文链接
http://www.eoeandroid.com/blog-531377-3446.html

你可能感兴趣的:(android,lauchmode)