Android笔记(二) | Activity的启动模式

启动模式

  在这里,首先要提到一个名词——任务栈(Task),数据结构中的栈我们都很熟悉,而Android系统采用栈的结构来管理应用程序运行过程中所启动的Activity,即任务栈。知道这一点,对于我们理解启动模式已经足够了。

  • standard(标准模式):系统的默认启动模式,每次启动一个Activity,都会重新创建一个实例,无论这个实例是否已经被创建。
     通常,当我们使用显式Intent启动一个standard模式的Activity时:
Intent intent = new Intent(MainActivity.this, NextActivty.class);
startActivity(intent);

NextActivity会进入MainActivity所在的栈中,即被启动的Activity会进入启动它的那个Activity所在的任务栈中。

  1. 这是一种典型的多实例模式,一个任务栈中可以有多个实例,每个实例也可以属于不同的任务栈。
  2. 注意:当我们用ApplicationContext启动一个standard模式的Activity时会报错,这是因为standard模式的Activity会默认进入启动它的Activity所属的任务栈中,但是非Activity类型的Context(如:ApplicationContext)没有所谓的任务栈。
  • singleTop(栈顶复用模式):在这种模式下,如果新Activity位于任务栈栈顶,就不会重新创建,而是回调它的 onNewIntent 方法;如果新Activity已经存在但不是位于栈顶,仍会重新创建新Activity。
  1. 例如:假设目前栈内情况为ABCD(D为栈顶),这时要再次启动D,如果D是singleTop模式,栈内情况不变;如果D是standard模式,栈内情况就会变为ABCDD。
  • singleTask(栈内复用模式):在这种模式下,只要Activity在一个栈中存在,多次启动这个Activity都不会重新创建实例,而是将这个Activity上面的Activity全部出栈,直到这个Activity位于栈顶;并且系统也会回调它的 onNewIntent 方法。
  1. 这是一种单实例模式,在一个任务栈中,一个Activity只会存在一个实例。
  2. 举例:
    1. 假设当前任务栈S1中的情况为ABCD,这时ActivityA以singleTask模式请求启动,其所需要的任务栈为S2,系统就会创建任务栈S2,然后创建A的实例入栈S2,任务栈S1不变。
    2. 假设A所需的任务栈为S1,其余情况同上,那么系统就会将BCD全部出栈,然后调用A的 onNewIntent 方法,这时任务栈S1内情况为A。
  • singleInstance(单实例模式):这种模式在singleTask模式的基础上,要求通过这种模式创建的Activity只能单独位于一个任务栈中,并且这个任务栈中只允许有一个实例存在。

上面是关于四种启动模式的概述,接下来再说明一些特殊情况:

  1. 假设目前有两个任务栈,前台任务栈的情况是AB,后台任务栈的情况是CD,假设CD的启动模式均为singleTask。现在请求启动C,由于后台任务栈中C位于栈底,所以C上面的Activity会被出栈,这时前台任务栈的情况为C,后台任务栈的情况为AB(过程见下图1.1)。
    Android笔记(二) | Activity的启动模式
    图1.1

    如果上面请求启动D,那么整个后台任务栈会切换到前台,前台任务栈会被切换到后台(过程见下图1.2)。
    Android笔记(二) | Activity的启动模式
    图1.2

补充:

  1. 为Activity指定所需任务栈的方式:
    在AndroidMenifest文件下,对应的activity标签内,有一个TaskAffinity属性,在默认情况下(即不指定TaskAffinity属性值),所有activity所需任务栈的名字为应用的包名。若我们要为某个Activity指定所需任务栈,只需指定TaskAffinity属性值和包名不同即可,这个值会作为这个Activity所在任务栈的名字,当然,这个Activity的启动模式为singleTask时才有意义。
<activity android:name=".SecondActivity"
    android:launchMode="singleTask"
    android:taskAffinity="com.example.laughter.task_1"/>
  1. 为Activity指定启动模式的方式:
    1. 在AndroidManifest文件中指定 launchMode 属性值。
    <activity android:name=".SecondActivity"
        android:launchMode="singleTask"
        android:taskAffinity="com.example.laughter.task_1"/>
    
    1. 通过在Intent中设置标志位来指定启动模式。(优先级高于第一种,两种方式同时存在时,以第二种为准。)
    Intent intent = new Intent(MainActivity.this, SecondActivity.class);
    intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
    startActivity(intent);
    

Activity的常见Flags(标志位)
  Activity的Flags有很多,有的可以设定Activity的启动模式,有的可以影响Activity的运行状态,在使用时需要注意有些标志位是系统内部使用的,应用程序不需要去手动设置这些标志位以防出现问题。

  1. FLAG_ACTIVITY_NEW_TASK
     为Activity指定“singleTask”启动模式
  2. FLAG_ACTIVITY_SINGLE_TOP
     为Activity指定“singleTop”启动模式
  3. FLAG_ACTIVITY_CLEAR_TOP
     具有此标志位的Activity,当它启动时,在同一个任务栈中所有位于它上面的Activity都要出栈。若该Activity为standard启动模式,那么它自身也会出栈,系统会重新创建这个Activity的实例放入栈顶。根据前面的内容可知,singleTask启动模式默认具有此标记位的效果。
上一篇:JVM笔记 | Java内存管理


下一篇:这是个测试