Activity启动模式

在Android应用中, Activity是最核心的组件, 如何生成一个Activity实例, 可以选择不同的启动模式, 即LaunchMode. 启动模式主要包括: standard, singleTop, singleTask, singleInstance.

标准模式在每次启动时, 都会创建实例; 三种单例模式, 会根据情况选择创建还是复用实例. 在Activity启动中, 创建实例的生命周期: onCreate -> onStart -> onResume; 重用实例的生命周期: onNewIntent -> onResume.

在AndroidManifest的Activity中, 使用launchMode属性, 可以设置启动模式, 默认是standard模式; 使用taskAffinity属性, 并添加包名, 可以设置Activity栈, 默认是当前包名, 只能应用于single模式.

希望通过本文, 可以更好的理解Activity的启动模式(LaunchMode).

Android

观察Activity栈的脚本,参考第5点.

adb shell dumpsys activity | sed -n-e'/Stack #/p'-e'/Running activities/,/Run #0/p'

本文示例的GitHub下载地址

1. Standard

标准模式, 启动Activity的默认模式,被启动的Activity会运行于启动的Activity栈, 因此必须使用Activity的Context启动, 不能使用Application, 否则会报错.

如MainActivity启动TestAActivity.

Stack #1:    Running activities (most recent first):      TaskRecord{3caa65e3#2711 A=me.chunyu.spike.wcl_activity_launchmode_demoU=0sz=2}        Run #1: ActivityRecord{36b06e99 u0 me.chunyu.spike.wcl_activity_launchmode_demo/.TestAActivityt2711}        Run #0: ActivityRecord{27396226u0 me.chunyu.spike.wcl_activity_launchmode_demo/.MainActivityt2711}  Stack #0:    Running activities (most recent first):      TaskRecord{27d796c9#2695 A=com.miui.homeU=0sz=1}        Run #0: ActivityRecord{2e5712cb u0 com.miui.home/.launcher.Launchert2695}

栈内由下到上: MainActivity -> TestAActivity.

2. SingleTop

栈顶复用模式. 只有Activity位于栈顶, 重复启动时, 会使用默认实例, 即单例模式; 如果位于栈内, 则仍然会创建实例.

MainActivity启动TestA, TestA启动TestB, TestB启动自身, TestB是单例. 观察栈内情况, TestB只有一份实例, 第二次创建复用.

Stack #1:    Running activities (most recent first):      TaskRecord{12abf566#2712 A=me.chunyu.spike.wcl_activity_launchmode_demoU=0sz=3}        Run #2: ActivityRecord{187d7ff7 u0 me.chunyu.spike.wcl_activity_launchmode_demo/.TestBActivityt2712}        Run #1: ActivityRecord{a551034 u0 me.chunyu.spike.wcl_activity_launchmode_demo/.TestAActivityt2712}        Run #0: ActivityRecord{22f9cce4 u0 me.chunyu.spike.wcl_activity_launchmode_demo/.MainActivityt2712}

栈内: MainActivity -> TestAActivity -> TestBActivity

MainActivity启动TestA, TestA启动TestB, TestB启动TestC, TestC启动TestB, TestB是单例. 观察栈内情况, 由于TestC是栈顶, TestC启动TestB, TestB不是栈顶, 重新创建TestB实例, 则保留两份TestB.

Stack #1:    Running activities (most recent first):      TaskRecord{1792f5f0#2715 A=me.chunyu.spike.wcl_activity_launchmode_demoU=0sz=5}        Run #4: ActivityRecord{1e70110b u0 me.chunyu.spike.wcl_activity_launchmode_demo/.TestBActivityt2715}        Run #3: ActivityRecord{c7f4dce u0 me.chunyu.spike.wcl_activity_launchmode_demo/.TestCActivityt2715}        Run #2: ActivityRecord{254536cd u0 me.chunyu.spike.wcl_activity_launchmode_demo/.TestBActivityt2715}        Run #1: ActivityRecord{36b2da15 u0 me.chunyu.spike.wcl_activity_launchmode_demo/.TestAActivityt2715}        Run #0: ActivityRecord{3a1c4a6a u0 me.chunyu.spike.wcl_activity_launchmode_demo/.MainActivityt2715}

栈内: MainActivity -> TestAActivity -> TestBActivity ->

TestCActivity -> TestBActivity

3. SingleTask

栈内复用模式, 只要Activity在一个栈中存在, 多次调用时, 都不会创建实例, 即单例模式.

情况包含以下几种:

(1) 任务栈不存在, 初次启动SingleTask实例, 会创建任务栈和实例.

MainActivity启动TestA, TestA启动TestB, TestB是SingleTask, 并且任务栈不同. 观察可知, 系统包含两个任务栈, TestB位于其他任务栈中.

Stack #1:    Running activities (most recent first):      TaskRecord{d5d53d4#2727 A=me.chunyu.spike.stackU=0sz=1}        Run #2: ActivityRecord{1d720e55 u0 me.chunyu.spike.wcl_activity_launchmode_demo/.TestBActivityt2727}      TaskRecord{a3f797d#2726 A=me.chunyu.spike.wcl_activity_launchmode_demoU=0sz=2}        Run #1: ActivityRecord{ffd689d u0 me.chunyu.spike.wcl_activity_launchmode_demo/.TestAActivityt2726}        Run #0: ActivityRecord{192310ac u0 me.chunyu.spike.wcl_activity_launchmode_demo/.MainActivityt2726}

使用taskAffinity属性, 添加新的Activity栈, 与SingleTask配合使用, Standard模式无效.

新任务栈是me.chunyu.spike.stack.

(2) 任务栈存在, 初次启动SingleTask实例, 会直接入栈, 与Standard模式相同.

(3) 任务栈相同, 再次启动SingleTask实例, 实例会置于栈顶, 并清除其上面实例, 具有clearTop的效果.

MainActivity启动TestA, TestA启动TestB, TestB是SingleTask, TestB启动TestC, TestC重新启动TestB, 则TestC会出栈. 观察可知, TestC出栈, TestB位于栈顶.

Stack #1:    Running activities (most recent first):      TaskRecord{18230815#2737 A=me.chunyu.spike.wcl_activity_launchmode_demoU=0sz=3}        Run #4: ActivityRecord{1126c300 u0 me.chunyu.spike.wcl_activity_launchmode_demo/.TestBActivityt2737}        Run #3: ActivityRecord{3114fee8 u0 me.chunyu.spike.wcl_activity_launchmode_demo/.TestAActivityt2737}        Run #2: ActivityRecord{f8e235d u0 me.chunyu.spike.wcl_activity_launchmode_demo/.MainActivityt2737}

TestC启动TestB, SingleTask模式, 导致clearTop, TestC出栈.

(4) 任务栈不同, 再次启动SingleTask实例, 会导致任务栈切换, 后台置于前台.

这比较难理解.

MainActivity启动TestA, TestA启动TestB(SingleTask实例, 不同任务栈), TestB启动TestC(与B类似), 则MainActivity和TestA相同栈, TestB和TestC相同栈, 此时栈顶是TestC. 按Home键, 再次启动应用, 则默认任务栈会启动, TestA启动, TestA启动TestC. 应用当前状态如下.

Stack #1:    Running activities (most recent first):      TaskRecord{1d05e6c9#2754 A=me.chunyu.spike.stackU=0sz=2}        Run #4: ActivityRecord{3f77e822 u0 me.chunyu.spike.wcl_activity_launchmode_demo/.TestCActivityt2754}      TaskRecord{3fe736d0#2753 A=me.chunyu.spike.wcl_activity_launchmode_demoU=0sz=2}        Run #3: ActivityRecord{15f0470e u0 me.chunyu.spike.wcl_activity_launchmode_demo/.TestAActivityt2753}      TaskRecord{1d05e6c9#2754 A=me.chunyu.spike.stackU=0sz=2}        Run #2: ActivityRecord{181229e6 u0 me.chunyu.spike.wcl_activity_launchmode_demo/.TestBActivityt2754}      TaskRecord{3fe736d0#2753 A=me.chunyu.spike.wcl_activity_launchmode_demoU=0sz=2}        Run #1: ActivityRecord{28628d61 u0 me.chunyu.spike.wcl_activity_launchmode_demo/.MainActivityt2753}      TaskRecord{2d646058#2719 A=com.android.incalluiU=0sz=1}

TestC至于栈顶, 点击Back键, 不是返回TestA(启动TestC的实例), 而是TestB, 即优先返回相同栈的实例. 再次是TestA, 然后是MainActivity, 依次出栈.

4. SingleInstance

单实例模式, 启动时, 系统会为其创造一个单独的任务栈, 以后每次使用, 都会使用这个单例, 直到其被销毁, 属于真正的单例模式.

示例: MainActivity启动TestA, TestA启动TestB(SingleInstance模式),

TestB启动TestC, TestC再启动TestB, 则仍启动上一次的TestB,

TestC合并入默认栈(MainActivity+TestA).

Stack #1:    Running activities (most recent first):      TaskRecord{384e3928#2765 A=me.chunyu.spike.wcl_activity_launchmode_demoU=0sz=1}        Run #3: ActivityRecord{1ffc5b6b u0 me.chunyu.spike.wcl_activity_launchmode_demo/.TestBActivityt2765}      TaskRecord{2ad03544#2764 A=me.chunyu.spike.wcl_activity_launchmode_demoU=0sz=3}        Run #2: ActivityRecord{293d8c37 u0 me.chunyu.spike.wcl_activity_launchmode_demo/.TestCActivityt2764}        Run #1: ActivityRecord{158bc0f3 u0 me.chunyu.spike.wcl_activity_launchmode_demo/.TestAActivityt2764}        Run #0: ActivityRecord{77691cf u0 me.chunyu.spike.wcl_activity_launchmode_demo/.MainActivityt2764}

5. startActivityForResult

Thx@回调的幸福时光

startActivityForResult不同于startActivity, 使用LaunchMode模式启动Activity时, 也会有一些不同, 可以正常传递数据, 但是无法连续创建自己时, 会生成多份实例.

TestB(singleTask模式)使用startActivity创建自己时, 会使用默认实例, 即单例; 而使用startActivityForResult创建自己时, 会生成一份新的示例.

Stack #1:    Running activities (most recent first):      TaskRecord{323200ac#2786 A=me.chunyu.clwang.stackU=0sz=3}        Run #4: ActivityRecord{3f9e14f3 u0 me.chunyu.spike.wcl_activity_launchmode_demo/.TestBActivityt2786}        Run #3: ActivityRecord{30d8f17b u0 me.chunyu.spike.wcl_activity_launchmode_demo/.TestBActivityt2786}        Run #2: ActivityRecord{11b95b5c u0 me.chunyu.spike.wcl_activity_launchmode_demo/.TestBActivityt2786}      TaskRecord{c86e175#2785 A=me.chunyu.spike.wcl_activity_launchmode_demoU=0sz=2}        Run #1: ActivityRecord{3558d7c4 u0 me.chunyu.spike.wcl_activity_launchmode_demo/.TestAActivityt2785}        Run #0: ActivityRecord{1b8620c u0 me.chunyu.spike.wcl_activity_launchmode_demo/.MainActivityt2785}

由此可知, 因为startActivityForResult需要返回值, 会保留实例, 覆盖单例效果.

注意: 4.x版本通过startActivityForResult启动singleTask, 无法正常获取返回值,参考.

5.x以上版本修复此问题, 考虑兼容性, 不推荐使用startActivityForResult和singleTask.

6. Intent设置标志位

Intent可以设置启动标志位, 即Flag.

intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);

AndroidManifest无法设置FLAG_ACTIVITY_CLEAR_TOP, 即清除栈上其他实例; Intent无法设置singleInstance启动模式. 两者选其一使用即可, 如都使用, Intent的优先级大于AndroidManifest的优先级.

常用的标志位:

FLAG_ACTIVITY_NEW_TASK: 同singleTask启动模式.

FLAG_ACTIVITY_SINGLE_TOP: 同singleTop启动模式.

FLAG_ACTIVITY_CLEAR_TOP: 一般和singleTask启动模式出现. 如果是singleTask启动模式, 会清除栈上其他实例, 复用实例, 调用onNewIntent; 如果是standard启动模式, 即默认模式, 则会清除自己和其他实例, 并重新创建, 调用 onCreate.

启动模式做为Activity的重要属性, 还是需要比较透彻的掌握.

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,761评论 5 460
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,953评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,998评论 0 320
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,248评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,130评论 4 356
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,145评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,550评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,236评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,510评论 1 291
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,601评论 2 310
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,376评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,247评论 3 313
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,613评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,911评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,191评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,532评论 2 342
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,739评论 2 335

推荐阅读更多精彩内容