Android 开发艺术探索读书笔记 1 -- Activity 的生命周期和启动模式

本篇文章主要介绍以下几个知识点:

  • Activity 的生命周期全面分析;
  • Activity 的启动模式。
hello,夏天 (图片来源于网络)

1.1 Activity 的生命周期全面分析

典型情况下的生命周期,指在有用户参与的情况下,Activity 所经过的生命周期的改变。

异常情况下的生命周期,指 Activity 被系统回收或由于设备的 Configuration 改变导致 Activity 被销毁重建。

1.1.1 典型情况下的生命周期分析

Activity 生命周期的切换过程

(1)针对一个特定的 Activity,首次启动,回调如下:onCreate ->onStart -> onResume

(2)当用户打开新的 Activity 或切换到桌面时,回调如下:onPause -> onStop。(注:若新 Activity 采用了透明主题,则当前 Activity 不会回调 onStop

(3)当用户再次回到原 Activity 时,回调如下:onRestart -> onStart -> onResume

(4)当用户按 back 键回退时,回调如下:onPause -> onStop -> onDestroy

(5)当 Activity 被系统回收后再次打开,生命周期方法回调过程和(1)一样。(注:只是生命周期方法一样,不代表所有过程都一样

(6)整个生命周期:onCreateonDestroy 是配对的(标识着 Activity 的创建和销毁,只调用一次)。
  Activity 是否可见:onStartonStop 是配对的(可能被调用多次);
  Activity 是否在前台:onResumeonPause 是配对的(可能被调用多次)。

问题 1:onStartonResumeonPauseonStop 从描述上看差不多,对我们来说有什么实质的不同呢?
  答:这两配对的回调具有不同的意义,onStartonStop 是根据 Activity 是否可见来回调的,而 onResumeonPause 是根据 Activity 是否位于前台来回调的,除此之外,在实际使用中无其他明显区别。

问题2:假设当前Activity为A,若这时用户打开一个新Activity B,那么B的 onResume 和 A 的 onPause 哪个先执行呢?
  答:当新启动一个 Activity 时,旧 Activity 的 onPause 会先执行,然后才会启动新的 Activity。(注:onPauseonStop 都不能执行耗时的操作,尤其是 onPause

1.1.2 异常情况下的生命周期分析

  • 情况 1:资源相关的系统配置发生改变导致 Activity 被杀死并重新创建

在默认情况下,若 Activity 不做特殊处理,当系统配置发生改变后,Activity 就会被销毁并重新创建,其生命周期如图:

异常情况下 Activity 的重建过程

当系统配置发生改变后,Activity 会被销毁,其 onPauseonStoponDestroy 均会被调用,同时由于 Activity 是在异常情况下终止的,系统会调用 onSaveInstanceState 来保存当前Activity 的状态。

方法 onSaveInstanceState 的调用时机是在 onStop 之前(和 onPause 无既定的时序关系),并且只会在 Activity 被异常终止的情况下回调

当 Activity 被重新创建后,系统会调用 onRestoreInstanceState,并把 Activity 销毁时 onSaveInstanceState 方法保存的 Bundle 对象作为参数同时传递给 onRestoreInstanceStateonCreate 方法。

因此,可以通过 onRestoreInstanceStateonCreate 方法来判断 Activity 是否被重建了,若被重建了,可取出之前保存的数据并恢复。(onRestoreInstanceState 在 onStart 之后调用

关于保存和恢复 View 层次结构,其工作流程为:首先 Activity 被意外终止时,Activity 会调用 onSaveInstanceState 去保存数据,然后 Activity 会委托 Window 去保存数据,接着 Window 再委托它上面的顶级容器去保存数据。

  • 情况 2:资源内存不足导致低优先级的 Activity 被杀死

Activity 按照优先级从高到低,可以分为如下三种:

(1)前台 Activity——正在和用户交互的 Activity,优先级最高。

(2)可见但非前台 Activity —— 如 Activity 中弹出一个对话框,导致 Activity 可见但位于后台无法和用户直接交互。

(3)后台 Activity——已经被暂停的 Activity,如执行了 onStop,优先级最低。

当系统内存不足时,系统会按上述优先级杀死目标 Activity 所在的进程,并在后续通过 onSaveInstanceStateonRestoreInstanceState 来存储和恢复数数据。

系统配置中有很多内容,当某项内容改变后,若不想系统重新创建 Activity,可以给 Activity 指定 configChanges 属性。

如给 configChanges 属性添加 android:configChanges="orientation" 可避免 Activity 在屏幕旋转时重新创建。

若想指定多个值,可用“|”连接起来,如 android:configChanges="orientation|keyboardHidden"

系统配置中所含的项目很多,如下:

configChanges 的项目和含义

上表中项目很多,但常用的只有 localeorientationkeyboardHidden 这三个。

值得注意的是 ,screenSizesmallestScreenSize 比较特殊,它们的行为和编译选项有关,和运行环境无关。

1.2 Activity 的启动模式

1.2.1 Activity 的 LaunchMode

(1)standard:标准模式。

系统的默认模式,每次启动一个 Activity 都会重新创建一个新的实例,不管这个实例是否已存在。

值得注意的是,用 ApplicationContext 去启动 standard 模式的 Activity 时会报错,如以下代码:

 tv_text.setOnClickListener {
     // 点击跳转到 KotlinActivity
     val intent = Intent()
     intent.setClass(applicationContext, KotlinActivity::class.java)
     applicationContext.startActivity(intent)
 }

运行会报如下错误:

standard 模式下用ApplicationContext 启动 Activity 报错

这是因为 standard 模式的 Activity 默认会进入启动它的 Activity 所属的任务栈中,但由于非 Activity 类型的 Context(如 ApplicationContext)并无所谓的任务栈,从而报错。

解决上面问题的方法是为待启动 Activity 指定 FLAG_ACTIVITY_NEW_TASK 标记位,这样启动时就会为它创建一个新的任务栈(此时待启动 Activity 是以 singleTask 模式启动的)。

(2)singleTop:栈顶复用模式。

此模式下,若新 Activity 已经位于任务栈的栈顶,则此 Activity 不会被重新创建,同时它的 onNewIntent 方法会被回调,通过此方法的参数可以取出当前请求的信息。

值得注意的是,这个 Activity 的 onCreateonStart 不会被系统调用,因为它并没有发生改变。

(3)singleTask:栈内复用模式。

一种单实例模式,此模式下,只要 Activity 在一个栈中存在,那么多次启动此 Activity 都不会重新创建实例,和 singleTop 一样,系统也会回调其 onNewIntent

(4)singleInstance:单实例模式。

一种加强的 singleTask 模式,具有 singleTask 模式的所有特性,并且具有此种模式的 Activity 只能单独地位于一个任务栈中。


给 Activity 指定启动模式有两种方法:

1. 通过 AndroidMenifest 为 Activity 指定启动模式

2. 通过在 Intent 中设置标志位为 Activity 指定启动模式。

二者区别在于:

(1) 优先级上,方式2的优先级高于方式1,当两种同时存在时,以方式2为准;

(2) 限定范围不同,比如,方式1无法直接为 Activity 设定 FLAG_ACTIVITY_CLEAR_TOP 标识,而方式2无法为 Activity 指定 singleInstance 模式。


1.2.2 Activity 的 Flags

Activity 的 Flags 有很多,有的标记位可以设定 Activity 的启动模式,有的会影响 Activity 的运行状态等。常用的如下:

  • FLAG_ACTIVITY_NEW_TASK

其作用是为 Activity 指定 singleTask 启动模式,效果和在 XML 中指定该模式相同。

  • FLAG_ACTIVITY_SINGLE_TOP

其作用是为 Activity 指定 singleTop 启动模式,效果和在 XML 中指定该模式相同。

  • FLAG_ACTIVITY_CLEAR_TOP

具有此标记位的 Activity 启动时,在同一个任务栈中所有位于它上面的 Activity 都要出栈。

此模式一般需要和 FLAG_ACTIVITY_NEW_TASK 配合使用,若被启动Activity 的实例已存在,则系统会调用它的 onNewIntent

  • FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS

具有此标记的 Activity 不会出现在历史 Activity 的列表中(用于某些情况不希望用户通过历史列表回到 Activity 时)。它等同于在 XML 中指定 Activity 的属性 android:excludeFromRecents="true"

本篇文章就介绍到这。

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

推荐阅读更多精彩内容