Activity启动模式的示例分析(activity,移动开发)

时间:2024-05-04 12:22:21 作者 : 石家庄SEO 分类 : 移动开发
  • TAG :

    Activity%E5%90%AF%E5%8A%A8%E6%A8%A1%E5%BC%8F%E7%9A%84%E7%A4%BA%E4%BE%8B%E5%88%86%E6%9E%90

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

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

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

观察Activity栈的脚本.

Standard

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

如MainActivity启动TestAActivity.

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

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

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

栈内: MainActivity -> TestAActivity -> TestBActivity

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

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

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

情况包含以下几种:
(1) 任务栈不存在, 初次启动SingleTask实例, 会创建任务栈和实例.

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

使用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位于栈顶.

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

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

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

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

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

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

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

startActivityForResult

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

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

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

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

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

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

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.
显示栈的Shell命令

Shell命令

直接获取Activity信息有些冗余, 我们只关注堆栈信息即可.

sed可以编辑显示的文字.

-n , 从截取处开始连续处理.

-e , 多选参数.

'/Stack #/p' , 输出含有 Stack # 的行.

-e '/Running activities/,/Run #0/p' , 输出从 Running activities 至 Run #0 的所有行.

输出结果

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

本文:Activity启动模式的示例分析的详细内容,希望对您有所帮助,信息来源于网络。
上一篇:Python+OpenCV如何实现角度测量下一篇:

5 人围观 / 0 条评论 ↓快速评论↓

(必须)

(必须,保密)

阿狸1 阿狸2 阿狸3 阿狸4 阿狸5 阿狸6 阿狸7 阿狸8 阿狸9 阿狸10 阿狸11 阿狸12 阿狸13 阿狸14 阿狸15 阿狸16 阿狸17 阿狸18