Android Activity启动模式详解

https://rainmonth.github.io/posts/A180505.html

Android Activity启动模式详解

四种启动模式分别是

  • standard
  • signleTop
  • singleTask
  • singleInstance

设置方式

  1. 在AndroidManifest.xml文件中配置,设置示例:

    1
    2
    3
    4
    <activity 
    android:name="com.rainmonth.MainActivity"
    android:launchMode="singleInstance" />

  2. 在代码中动态的设置,相关的Activity Flag有:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    49
    50
    51
    52
    53
    54
    55
    56
    57
    58
    59
    60
    61
    62
    63
    64
    65
    66
    67
    /**
    * 设置后,不会记录在Activity Stack,离开就finish()掉了
    */
    Intent.FLAG_ACTIVITY_NO_HISTORY;

    /**
    * 设置后,如果该Activity在栈顶,则该Activity不再重新启动
    */
    Inent.FLAG_ACTIVITY_SINGLE_TOP;

    /**
    * 比较复杂,后面会详细说明
    */
    Intent.FLAG_ACTIVITY_NEW_TASK;

    /**
    * 设置后,会创建一个Task,并在这个Task中启动一个Activity,该Flag要配置
    * NEW_TASK或者NEW_DOCUMENT,来使用单独使用会被忽略,该Flag要慎用(除非
    * 你实现了应用层Launcher)
    */
    Intent.FLAG_ACTIVITY_MULTIPLE_TASK;

    /**
    * 设置后,且该Activity已经运行于当前task中,那么Activity栈中位于该
    * Activity之上的都将关闭,该Activity将视传递过来的启动自己的intent为
    * 新的intent,接受的方法为onNewIntent(),通常配
    * Intent.FLAG_ACTIVITY_NEW_TASK来用于通知的管理中
    */
    Intent.FLAG_ACTIVITY_CLEAR_TOP;

    /**
    * 设置后,且该Intent是用于启动一个已经有实例存在与Activity栈的
    * Activity新实例,那么接受该Activity返回结果的目标将会被传递到这个
    * Activity新实例,也就是说这个Activity可以通过setResult()方法来将结果
    * 返回到之前接受返回结果的目标,通俗讲就是新来的也可以和老的上级交互
    */
    Intent.FLAG_ACTIVITY_FORWARD_RESULT;

    Intent.FLAG_ACTIVITY_PREVIOUS_IS_TOP;

    Intent.FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS;

    Intent.FLAG_ACTIVITY_RETAIN_IN_RECENTS;

    Intent.FLAG_ACTIVITY_BROUGHT_TO_FRONT;

    Intent.FLAG_ACTIVITY_RESET_TASK_IF_NEEDED;

    Intent.FLAG_ACTIVITY_LAUNCHED_FROM_HISTORY;

    Intent.FLAG_ACTIVITY_CLEAR_WHEN_TASK_RESET;

    Intent.FLAG_ACTIVITY_NEW_DOCUMENT;

    Intent.FLAG_ACTIVITY_NO_USER_ACTION;

    Intent.FLAG_ACTIVITY_REORDER_TO_FRONT;

    /**
    * 设置后,防止在Activity上应用动画
    */
    Intent.FLAG_ACTIVITY_NO_ANIMATION;

    Intent.FLAG_ACTIVITY_CLEAR_TASK;

    Intent.FLAG_ACTIVITY_TASK_ON_HOME;

设置示例:

1
2
3
4
5
6
Intent intent = new Intent(this, XxxActivity.class);
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
intent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TASK);
intent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
...
startActivity(intent)

四种模式的作用

standard

默认模式,可以不用写配置。在这个模式下,都会默认创建一个新的实例。因此,在这种
模式下,可以有多个相同的实例,也允许多个相同Activity叠加。例如:

1
2
3
若我有一个Activity名为A1, 上面有一个按钮可跳转到A1。那么如果我点击按钮,
便会新启一个Activity A1叠在刚才的A1之上,再点击,又会再新启一个在它之上……
点back键会依照栈顺序依次退出。

singleTop

可以有多个实例,但是不允许多个相同Activity叠加。即,如果Activity在栈顶的时候,
启动相同的Activity,不会创建新的实例,而会调用其onNewIntent方法。例如:

1
2
3
4
5
6
7
8
若我有两个Activity名为B1,B2,两个Activity内容功能完全相同,都有两个按钮可
以跳到B1或者B2,唯一不同的是B1为standard,B2为singleTop。

若我意图打开的顺序为B1->B2->B2,则实际打开的顺序为B1->B2(
后一次意图打开B2,实际只调用了前一个的onNewIntent方法);

若我意图打开的顺序为B1->B2->B1->B2,则实际打开的顺序与意图的一致,
为B1->B2->B1->B2。

singleTask

只有一个实例。在同一个应用程序中启动他的时候,若Activity不存在,则会在当前task
创建一个新的实例,若存在,则会把task中在其之上的其它Activity destroy掉并调用
它的onNewIntent方法。如果是在别的应用程序中启动它,则会新建一个task,并在该
task中启动这个Activity,singleTask允许别的Activity与其在一个task中共存,
也就是说,如果我在这个singleTask的实例中再打开新的Activity,这个新的
Activity还是会在singleTask的实例的task中。例如:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
若我的应用程序中有三个Activity,C1,C2,C3,三个Activity可互相启动,其中
C2为singleTask模式,那么,无论我在这个程序中如何点击启动,如:
C1->C2->C3->C2->C3->C1-C2,C1,C3可能存在多个实例,但是C2只会存在
一个,并且这三个Activity都在同一个task里面。
但是C1->C2->C3->C2->C3->C1-C2,这样的操作过程实际应该是如下这样的,
因为singleTask会把task中在其之上的其它Activity destory掉。
操作:
C1->C2
C1->C2->C3
C1->C2->C3->C2
C1->C2->C3->C2->C3->C1
C1->C2->C3->C2->C3->C1-C2
实际:
C1->C2
C1->C2->C3
C1->C2
C1->C2->C3->C1
C1->C2

若是别的应用程序打开C2,则会新启一个task。如别的应用Other中有一个
activity,taskId为200,从它打开C2,则C2的taskIdI不会为200,例如C2的
taskId 为201,那么再从C2打开C1、C3,则C2、C3的taskId仍为201。

注意:
如果此时你点击home,然后再打开Other,发现这时显示的肯定会是Other应用
中的内容,而不会是我们应用中的C1 C2 C3中的其中一个。

singleInstance

只有一个实例,并且这个实例独立运行在一个task中,这个task只有这个实例,不允许有
别的Activity存在。例如:

1
2
3
4
5
6
7
8
9
10
11
12
程序有三个Activity:D1,D2,D3,三个Activity可互相启动,其中D2为
singleInstance模式。那么程序从
D1开始运行,假设D1的taskId为200,那么从D1启动D2时,
D2会新启动一个task,即D2与D1不在一个task中
运行。假设D2的taskId为201,再从D2启动D3时,
D3的taskId为200,也就是说它被压到了D1启动的任务栈中。

若是在别的应用程序打开D2,假设Other的taskId为200,
打开D2,D2会新建一个task运行,假设它的taskId为
201,那么如果这时再从D2启动D1或者D3,则又会再创建
一个task,因此,若操作步骤为other->D2->D1,这过
程就涉及到了3个task了。

TaskAffinity属性

可以为Activity指定一个Task,然后配合Intent.FLAG_ACTIVITY_NEW_TASK来使用。这样Activity就在指定的Task中运行了。