×

深入理解Activity启动模式之launchMode

96
静默加载
2017.07.19 19:56* 字数 897

Android每个Application都是由若干个四大组件组成的。每个页面都是一个Activity,当需要打开相应页面(Activity)时系统会创建他们的实例并把他们一一放入栈中进行管理。任务栈是一种“后进先出”的栈结构,通过back键,我们可以发现这些Activity会一一出栈(PS:不断返回上一页)。如果每次启动Activity都创建一个实例,会不会很浪费资源。能不能进行Activity的复用呢?Android系统在设计就考虑到这个问题,所以提供了同步的Activity启动模式,在不同条件下进行Activity的复用。其中都包括:Standard、SingleTop、SingleTask、SingleInstance。
(PS:本篇文章的实验数据都基于Android7.0)

在讲述Activity启动模式之前我们先了解一些基础概念:

Application 是组件的集合。
Process 操作系统调度的单位。
Task是指在执行特定作业时与用户交互的一系列 Activity。Task以栈的形式管理Activity集合。

  • 一个应用中可以有多个Task
  • 不同应用的Activity可以放在同一个Task中进行管理
  • 一个应用中可以有多个Process
  • Task可以运行在不同进程

standard

标准启动模式,也是activity的默认启动模式。在这种模式下启动的activity可以被多次实例化,即在同一个任务中可以存在多个activity的实例,每个实例都会处理一个Intent对象。

standard.png
  • AndroidManifest.xml
<application
    android:allowBackup="true"
    android:icon="@mipmap/ic_launcher"
    android:label="@string/app_name"
    android:supportsRtl="true"
    android:theme="@style/AppTheme" >
    <activity android:name=".AActivity"
        android:launchMode="standard">
        <intent-filter>
            <action android:name="android.intent.action.MAIN" />

            <category android:name="android.intent.category.LAUNCHER" />
        </intent-filter>
    </activity>
    <activity android:name=".BActivity"
              android:launchMode="standard"/>
</application>
  • Logcat输出
3459-3459 D/$$$AActivity: onCreate: currentActivityName:AActivity currentTaskID:6384
3459-3459 D/$$$AActivity: onClickBtn: AActivity start BActivity
3459-3459 D/$$$BActivity: onCreate: currentActivityName:BActivity currentTaskID:6384
3459-3459 D/$$$BActivity: onClickBtn: BActivity start AActivity
3459-3459 D/$$$AActivity: onCreate: currentActivityName:AActivity currentTaskID:6384
3459-3459 D/$$$AActivity: onClickBtn: AActivity start BActivity
3459-3459 D/$$$BActivity: onCreate: currentActivityName:BActivity currentTaskID:6384
  • 当前任务栈
 Running activities (most recent first):
      TaskRecord{3efeab #6384 A=com.tzx.launchmodel U=0 StackId=1 sz=4}
        Run #4: ActivityRecord{18878b0 u0 com.tzx.launchmodel/.BActivity t6384}
        Run #3: ActivityRecord{5a7f957 u0 com.tzx.launchmodel/.AActivity t6384}
        Run #2: ActivityRecord{559bd5f u0 com.tzx.launchmodel/.BActivity t6384}
        Run #1: ActivityRecord{facb908 u0 com.tzx.launchmodel/.AActivity t6384}

singleTop

启动的activity的实例已经存在于任务桟的桟顶,那么再启动这个Activity时,不会创建新的实例,而是重用位于栈顶的那个实例,并且会调用该实例的onNewIntent()方法将Intent对象传递到这个实例中。

singleTop.png
  • AndroidManifest.xml
<application
    android:allowBackup="true"
    android:icon="@mipmap/ic_launcher"
    android:label="@string/app_name"
    android:supportsRtl="true"
    android:theme="@style/AppTheme" >
    <activity android:name=".AActivity"
        android:launchMode="standard">
        <intent-filter>
            <action android:name="android.intent.action.MAIN" />

            <category android:name="android.intent.category.LAUNCHER" />
        </intent-filter>
    </activity>
    <activity android:name=".BActivity"
              android:launchMode="singleTop"/>
</application>
  • Logcat输出
14450-14450 D/$$$AActivity: onCreate: currentActivityName:AActivity currentTaskID:6386
14450-14450 D/$$$AActivity: onClickBtn: AActivity start BActivity
14450-14450 D/$$$BActivity: onCreate: currentActivityName:BActivity currentTaskID:6386
14450-14450 D/$$$BActivity: onClickBtn: BActivity start AActivity
14450-14450 D/$$$AActivity: onCreate: currentActivityName:AActivity currentTaskID:6386
14450-14450 D/$$$AActivity: onClickBtn: AActivity start BActivity
14450-14450 D/$$$BActivity: onCreate: currentActivityName:BActivity currentTaskID:6386
14450-14450 D/$$$BActivity: onClickBtn: BActivity start BActivity
14450-14450 D/$$$BActivity: onNewIntent()
14450-14450 D/$$$BActivity: onClickBtn: BActivity start BActivity
14450-14450 D/$$$BActivity: onNewIntent()
  • 当前任务栈
Running activities (most recent first):
      TaskRecord{fba1a37 #6386 A=com.tzx.launchmodel U=0 StackId=1 sz=4}
        Run #5: ActivityRecord{9609e73 u0 com.tzx.launchmodel/.BActivity t6386}
        Run #4: ActivityRecord{50fff56 u0 com.tzx.launchmodel/.AActivity t6386}
        Run #3: ActivityRecord{1c403e9 u0 com.tzx.launchmodel/.BActivity t6386}
        Run #2: ActivityRecord{3c41ba4 u0 com.tzx.launchmodel/.AActivity t6386}

singleTask

启动模式为singleTask,那么系统总会在一个新任务的最底部(root)启动这个activity,并且被这个activity启动的其他activity会和该activity同时存在于这个新任务中。如果系统中已经存在这样的一个activity则会重用这个实例,并且调用他的onNewIntent()方法。即,这样的一个activity在系统中只会存在一个实例。

singleTask.png
  • AndroidManifest.xml
<application
    android:allowBackup="true"
    android:icon="@mipmap/ic_launcher"
    android:label="@string/app_name"
    android:supportsRtl="true"
    android:theme="@style/AppTheme" >
    <activity android:name=".AActivity"
        android:launchMode="standard">
        <intent-filter>
            <action android:name="android.intent.action.MAIN" />

            <category android:name="android.intent.category.LAUNCHER" />
        </intent-filter>
    </activity>
    <activity android:name=".BActivity"
              android:launchMode="singleTask"/>
    <activity android:name=".CActivity"
        android:process=":remote"/>
</application>
  • Logcat输出
19684-19684 D/$$$AActivity: onCreate: currentActivityName:AActivity currentTaskID:6394
19684-19684 D/$$$AActivity: onClickBtn: AActivity start BActivity
19684-19684 D/$$$BActivity: onCreate: currentActivityName:BActivity currentTaskID:6394
19684-19684 D/$$$BActivity: onClickBtn: BActivity start AActivity
19684-19684 D/$$$AActivity: onCreate: currentActivityName:AActivity currentTaskID:6394
19684-19684 D/$$$AActivity: onClickBtn: AActivity start CActivity
19921-19921 D/$$$CActivity: onCreate: currentActivityName:CActivity currentTaskID:6394
19921-19921 D/$$$CActivity: onClickBtn: CActivity start AActivity
19684-19684 D/$$$AActivity: onCreate: currentActivityName:AActivity currentTaskID:6394
19684-19684 D/$$$AActivity: onClickBtn: AActivity start BActivity
19684-19684 D/$$$BActivity: onNewIntent()
19684-19684 D/$$$BActivity: onClickBtn: BActivity start BActivity
19684-19684 D/$$$BActivity: onNewIntent()
  • 当前任务栈
Running activities (most recent first):
      TaskRecord{8e8fb09 #6394 A=com.tzx.launchmodel U=0 StackId=1 sz=2}
        Run #7: ActivityRecord{7489fc u0 com.tzx.launchmodel/.BActivity t6394}
        Run #6: ActivityRecord{9a5d10e u0 com.tzx.launchmodel/.AActivity t6394}

singleInstance

总是在新的任务中开启,并且这个新的任务中有且只有这一个实例。

singleInstance.png
  • AndroidManifest.xml
<application
    android:allowBackup="true"
    android:icon="@mipmap/ic_launcher"
    android:label="@string/app_name"
    android:supportsRtl="true"
    android:theme="@style/AppTheme" >
    <activity android:name=".AActivity"
        android:launchMode="standard">
        <intent-filter>
            <action android:name="android.intent.action.MAIN" />

            <category android:name="android.intent.category.LAUNCHER" />
        </intent-filter>
    </activity>
    <activity android:name=".BActivity"
              android:launchMode="singleInstance"/>
    <activity android:name=".CActivity"
        android:process=":remote"/>
</application>
  • Logcat输出
25016-25016 D/$$$AActivity: onCreate: currentActivityName:AActivity currentTaskID:6396
25016-25016 D/$$$AActivity: onClickBtn: AActivity start BActivity
25016-25016 D/$$$BActivity: onCreate: currentActivityName:BActivity currentTaskID:6397
25016-25016 D/$$$BActivity: onClickBtn: BActivity start AActivity
25016-25016 D/$$$AActivity: onCreate: currentActivityName:AActivity currentTaskID:6396
25016-25016 D/$$$AActivity: onClickBtn: AActivity start CActivity
26003-26003 D/$$$CActivity: onCreate: currentActivityName:CActivity currentTaskID:6396
26003-26003 D/$$$CActivity: onClickBtn: CActivity start BActivity
25016-25016 D/$$$BActivity: onNewIntent()
  • 当前任务栈
 Running activities (most recent first):
      TaskRecord{d5934c9 #6397 A=com.tzx.launchmodel U=0 StackId=1 sz=1}
        Run #9: ActivityRecord{986e7ce u0 com.tzx.launchmodel/.BActivity t6397}
      TaskRecord{f282078 #6396 A=com.tzx.launchmodel U=0 StackId=1 sz=3}
        Run #8: ActivityRecord{d2c75de u0 com.tzx.launchmodel/.CActivity t6396}
        Run #7: ActivityRecord{2d3a2eb u0 com.tzx.launchmodel/.AActivity t6396}
        Run #6: ActivityRecord{2ca9351 u0 com.tzx.launchmodel/.AActivity t6396}

本文叙述性的文字非常少,主要通过其代码和其运行结果然后得出我们需要的结论。每一种设计模式我都画了一副图,可以使大家理解记忆深刻。
Activity启动模式到这里就结束了么,然而并没有。之前看过其他类似文章或者书籍的同学都知道还有Activity的Flags和 android:taskAffinity 标签对Activity的启动有影响。这些内容我们放在下一张深入理解Activity启动模式之大结局中讲述,有兴趣的同学可以点击阅读~!

文章到这里就全部讲述完啦,若有其他需要交流的可以留言哦

想阅读作者的更多文章,可以查看我 个人博客 和公共号:

振兴书城

Android
Web note ad 1