对于activity的七个声生命周期回调,总是被大家翻来覆去的说,甚至说的都有些厌烦了,这部分知识虽然基础但也很重要,谁都不想在面试的时候只说出个一知半解,下面的分析是对阅读《安卓开发艺术探索》第一章后的整理和思考。

一、正常情况下的生命周期分析

先来上一张大家都熟悉的流程图,来复习一遍活动的生命周期

然后来简单说说这七个回调方法

  • onCreate : activity被创建时调用,一般在这个方法中进行活动的初始化工作,如设置布局文件、加载数据、绑定控件等。

  • onStart : 经历该回调后,activity由不可见变为可见,但此时处于后台可见,还不能和用户进行交互。

  • onResume : 已经可见的activity从后台来到前台,可以和用户进行交互。

  • onPause : 当用户启动了新的activity,原来的activity不再处于前台,也无法和用户进行交互,并且紧接着就会调用onStop()方法,但如果用户这时立刻按返回键回到原activity,就会调用onResume()方法让活动重新回到前台。而且在官方文档中给出了说明,不允许在onPause()方法中执行耗时操作,因为这会影响到新activity的启动,具体原因我们在后面分析。

  • onStop : 这个回调代表了activity由可见变为完全不可见,在这里可以进行一些稍微重量级的操作。需要注意的是,处于onPause()onStop()回调后的activity优先级很低,当有优先级更高的应用需要内存时,该应用就会被杀死,那么当再次返回原activity的时候,会重新调用activity的onCreate()方法。

  • onDestroy : 来到了这个回调,说明activity即将被销毁,应该将资源的回收和释放工作在该方法中执行。

  • onRestart : 这个回调代表了activity由完全不可见重新变为可见的过程,当activity经历了onStop()回调变为完全不可见后,如果用户返回原activity,便会触发该回调,并且紧接着会触发onStart()来使活动重新可见。

想必大家已经对这个过程非常熟悉了,下面我们通过一些实际的场景来更加深入的理解一下活动的启动流程。

1、 由活动A启动活动B时,活动A的onPause()与活动B的onResume()哪一个先执行?

下面创建两个正常的活动MainActivityFirstActivity,在MainActivity中设置按钮点击进入FirstActivity,看看会发生什么:

可以看到,是旧的Activity先执行onPause,新活动才开始启动。下面点击返回按钮:

点击返回后,同样是新Activity先执行onPause,旧的活动才开始重新启动,进行onRestart->onStart->onResume的流程,也就是说当发生活动切换时,是原活动先执行onPause,然后紧接着目标活动开始创建或重新启动。

2、dialog是否会对生命周期产生影响

从定义上来说,如果一个活动不在前台,也并非完全不可见,这个活动就会处在onPause()的暂停状态,我们来模拟一下这种情况,在MainActivity中设置三个按钮,第一个按钮点击后会弹出一个标准的AlertDialog,第二个按钮会弹出一个全屏的AlertDialog,第三个按钮点击会出现一个主题为Theme.AppCompat.Dialog的activity然后观察生命周期的变化:

首先可以看到,无论是正常的dialog还是全屏的dialog,活动依然维持在onResume()的状态,说明单纯的dialog并不会引起生命周期的变化。下面来看dialog主题的activity:

在启动DialogActivity后,原来的活动进入onPause(),新活动正常进行onCreate->onStart->onResume的流程,而原来的活动因为并没有完全不可见,所以也没有执行onStop,事实上除了dialog主题的活动,一些透明主题的活动也能达到同样的效果,接下来我们点击返回按钮:

由于MainActivity根本没有进入onStop的状态,所以返回时也不会进行onRestart->onStart的流程,而是直接onResume回到前台。

二、异常状态下活动的生命周期

当活动在运行过程中发生了某些异常情况时,上述所讨论的生命周期流程可能会受到影响,这里主要讨论两种异常情况。

1、资源配置改变导致activity重建

最常见的一种情况就是横竖屏的切换导致资源的变化,当程序启动时,会根据不同的配置加载不同的资源,例如横竖屏两个状态对应着两张不同的资源图片。如果在应用使用过程中屏幕突然旋转,那么activity就会因为系统配置发生改变而销毁重建,加载合适的资源。

(1) 数据保存

对于活动重新创建,我们如何保证activity中的已有数据不丢失呢,系统为我们提供了onSaveInstanceStateonRestoreInstanceState来保存和获取数据。

@Override
protected void onSaveInstanceState(Bundle outState) {
super.onSaveInstanceState(outState);
Log.i(TAG, "onSaveInstanceState: ");
} @Override
protected void onRestoreInstanceState(Bundle savedInstanceState) {
super.onRestoreInstanceState(savedInstanceState);
Log.i(TAG, "onRestoreInstanceState: ");
}

在活动异常销毁之前,系统会调用onSaveInstanceState,我们可以在Bundle类型的参数中保存想要的信息,之后这个Bundle对象会作为参数传递给onRestoreInstanceStateonCreat方法,这样在重新创建时就可以获取数据了。关于这两个方法,有几点需要注意的地方:

  • onSaveInstanceState方法的调用时机是在onStop之前,与onPause没有固定的时序关系。而onRestoreInstanceState方法则是在onStart之后调用。

  • 正常情况下的活动销毁并不会调用这两个方法,只有当活动异常销毁并且有机会重新展示的时候才会进行调用,除了资源配置的改变外,activity因内存不足被销毁也是通过这两个方法保存数据。

  • onRestoreInstanceStateonCreat都可以进行数据恢复工作,但是根据官方文档建议采用在onRestoreInstanceState中去恢复。

  • onRestoreInstanceStateonRestoreInstanceState这两个方法中,系统会默认为我们进行一定的恢复工作,例如EditText中的文本信息、ListView中的滚动位置等,下面对一些控件观察实际保存效果。

    • EditText:个人在对EditText实验的时候,发现转屏后文本信息并没有被保存,经过查询,发现了这样一句话:

      "Note: In order for the Android system to restore the state of the views in your activity, each view must have a unique ID, supplied by the android:id attribute." Android系统存储和还原View的状态必须有一个唯一的ID

      果然加上id之后EditText中的信息可以被自动保存了。

    • TextView:这里指的是通过setText方法动态设置文本内容,在这种情况下即使加了id也无法自动保存,这种情况可以通过给TextView设置freezesText属性才能自动保存,当然这条属性对EditText也同样适用。

(2) 防止重建

我们已经知道默认情况下,资源配置改变会导致活动的重新创建,但我们可以通过对活动android:configChanges属性的设置使活动防止重新被创建,我们来看看这个属性中有哪些内容:

属性值 含义
mcc SIM卡唯一标识IMSI(国际移动用户标识码)中的国家代码,由三位数字组成,中国为:460 这里标识mcc代码发生了改变
mnc SIM卡唯一标识IMSI(国际移动用户标识码)中的运营商代码,有两位数字组成,中国移动TD系统为00,中国联通为01,电信为03,此项标识mnc发生了改变
locale 设备的本地位置发生了改变,一般指的是切换了系统语言
touchscreen 触摸屏发生了改变
keyboard 键盘类型发生了改变,比如用户使用了外接键盘
keyboardHidden 键盘的可访问性发生了改变,比如用户调出了键盘
navigation 系统导航方式发生了改变
screenLayout 屏幕布局发生了改变,很可能是用户激活了另外一个显示设备
fontScale 系统字体缩放比例发生了改变,比如用户选择了个新的字号
uiMode 用户界面模式发生了改变,比如开启夜间模式-API8新添加
orientation 屏幕方向发生改变,比如旋转了手机屏幕
screenSize 当屏幕尺寸信息发生改变(当编译选项中的minSdkVersion和targeSdkVersion均低于13时不会导致Activity重启)API13新添加
smallestScreenSize 设备的物理屏幕尺寸发生改变,这个和屏幕方向没关系,比如切换到外部显示设备-API13新添加
layoutDirection 当布局方向发生改变的时候,正常情况下无法修改布局的layoutDirection的属性-API17新添加

如果不希望某些资源配置改变时活动被重建,只需在manifest中为相应活动添加属性即可,例如 configChanges="orientation"可以防止横竖屏引发的重启,然而事实上单加这条属性并没有什么效果,因为在api13之后,新添加的属性screenSize属性也会跟着设备的横竖切换而改变,所以正确的配置应该是configChanges="orientation|screenSize";而在api13之前,正确的配置应该是configChanges="orientation|keyboardHidden"

这里还要介绍一个重写方法onConfigurationChanged,用来监听资源配置的改变,这个方法只有在设置了configChanges并且相应的属性发生了变化时才会被调用,

@Override
public void onConfigurationChanged(Configuration newConfig) {
super.onConfigurationChanged(newConfig);
//监听横竖屏的变化
Log.i(TAG, "onConfigurationChanged: "+newConfig.orientation);
}

2、低优先级的activity由于内存不足被杀死

这种情况的数据保存方法和上一种情况相同,在这里简单说一下系统回收进程的优先级:

(1) 前台进程:
  • 持有用户正在交互的activity,即生命周期处于onResume状态的活动。
  • 该进程有绑定到正在交互的Activity的service或前台service。
(2) 可见进程:

这种进程虽然不在前台,但是仍然可见。

  • 该进程持有的Activity执行了onPause但未执行onStop。例如原活动启动了一个 dialog主题的activity,但此时原活动并非完全不可见。
  • 该进程有service绑定到可见的或前台Activity。
(3) 服务进程:

进程中持有一个service,同时不属于上面两种情况。

(4) 后台进程:

不属于上面三种情况,但进程持有一个不可见的activity,即执行了onStop但未执行onDestroy的状态。

(5) 空进程:

不包含任何活跃的应用组件,作用是加快下次启动这个进程中组件所需要的时间,优先级最低。

activity的启动模式

和生命周期一样,activity的四种launchMode也非常重要但又特别容易混淆,首先,activity是以任务栈的形式创建和销毁的,栈是一种“后进先出”的数据结构,在默认情况下,启动第一个activity时,系统将会为它创建一个任务栈并将活动置于栈底,而从这个activity启动的其他activity将会依次入栈,当用户连续按下返回键时,任务栈中的activity会从栈顶开始依次销毁。但是这样有一个弊端,就是对于某些activity我们不希望它总是重新创建,这时就需要采用不同的启动模式,下面就简单复习下activity的四种启动模式 :

  • standard(标准模式) : 这是activity的默认启动模式,只要启动activity就会创建一个新实例,例如有两个活动ActivityA和AciivityB,现在从活动A中连续3次启动B活动,那么活动B就会依次创建三个实例,按顺序进入ActivityA所在的任务栈中。

执行adb shell dumpsys activity命令观察任务栈中的实际情况:

  • singleTop(栈顶复用) :在这种启动模式下,首先会判断要启动的活动是否已经存在于栈顶,如果是的话就不创建新实例,直接复用栈顶活动。如果要启动的活动不位于栈顶或在栈中无实例,则会创建新实例入栈。例如栈中有活动A、B、C,启动模式全部为singleTop,现在想要新建一个活动C,执行完成后任务栈中的情况依然为A、B、C; 但是如果新建一个活动A,因为A不位于栈顶,所以会重新创建实例入栈,任务栈变为:A、B、C、A,

    初始任务栈状态

    接着启动活动A

    可以看到活动A被重新创建入栈,但如果是启动活动C,栈内活动不会改变,只不过活动C会先经历onPause,然后回调onNewIntent方法,紧接着执行onResume

  • singleTask(栈内复用) : 这种模式比较复杂,是一种栈内单例模式,当一个activity启动时,会进行两次判断

    • 首先会寻找是否有这个活动需要的任务栈,如果没有就创建这个任务栈并将活动入栈,如果有的话就进入下一步判断。

    • 第二次判断这个栈中是否存在该activity的实例,如果不存在就新建activity入栈,如果存在的话就直接复用,并且带有clearTop效果,会将该实例上方的所有活动全部出栈,令这个activity位于栈顶。

    场景一:假设当前任务栈中只有活动A,想要从A启动launchModesingleTask的活动B,但是活动B指定的任务栈与A不同,这里用到了TaskAffinity属性,相当于指定了想要的任务栈,下面会详细介绍。

      <activity
    android:name=".test.ActivityA"
    android:taskAffinity="com.example.a41061.task1">
    <intent-filter>
    <action android:name="android.intent.action.MAIN"/>
    <category android:name="android.intent.category.LAUNCHER"/>
    </intent-filter>
    </activity> <activity
    android:name=".test.ActivityB"
    android:launchMode="singleTask"
    android:taskAffinity="com.example.a41061.task2"/>

    启动后可以看到活动B运行在了一个新task中。

    场景二: 当前任务栈task1中存在活动A,从A中连续启动三个活动,顺序为B->C->B,B、C的启动模式均为singleTask,请求栈为task2,最后的启动结果将和上一种场景一样,不再重复展示,这里体现了singleTask模式的clearTop属性,第二次启动activityB后会复用栈底的实例,并将activityC出栈。

  • singleInstance(单例) : 这种模式是真正的单例模式,以这种模式启动的活动会单独创建一个任务栈,并且依然遵循栈内复用的特性,保证了这个栈中只能存在这一个活动。

还有一些需要注意的属性

  • **onNewIntent()**方法 : 后三种模式都会出现活动复用的情况,一旦活动被复用,就会回调用onNewIntent方法,通过这个方法中的Intent参数就可以进行页面的更新,举一个实际应用场景的例子:

    • 在活动A点击设置密码按钮进入活动B: A->B
    • 在活动B中设置密码后点击完成后进入活动C: A->B->C
    • 在活动C中点击确认,返回活动A,并且携带已经确认的信息: A->B->C->A
    • 活动A在onNewIntent方法中获取信息,将设置密码字样改为修改密码。
  • TaskAffinity属性 : 这个属性代表活动的亲和性,即一个活动启动时想要指定的任务栈名字,在默认情况下,所有活动所需的任务栈名字为所应用的包名。

参考文章

 

老生常谈-Activity(山东数漫江湖)的更多相关文章

  1. 教你 Shiro 整合 SpringBoot,避开各种坑(山东数漫江湖)

    依赖包 <dependency> <groupId>org.apache.shiro</groupId> <artifactId>shiro-sprin ...

  2. NodeJS 微信公共号开发 - 响应微信发送的Token验证(山东数漫江湖)

    背景 使用 NodeJS 进行微信公共号开发,首先需要响应微信发送的Token验证,官方文档 填写服务器配置 登录微信公共平台,在开发下的基本配置打开该页面. 依次填写接口的 URL.自定义的 Tok ...

  3. Web Session 浅入浅出(山东数漫江湖)

    使用过几种Web App开发语言和框架,都会接触到Session的概念.即使是一个简单站点访问计数的功能,也常常使用Session来实现的.其他常用的领域还有购物车,登录用户等.但是,对Session ...

  4. 炒鸡简单的canvas粒子(山东数漫江湖)

    位图的canvas一直不会被svg比下去的原因了. 俗话说,须弥芥子,是大小之说,也有以小见大之说,颗颗粒子,足以构建宏大效果. 这是一篇炒鸡简单的canvas粒子教程,主要是讲如何粒子特效的原理,一 ...

  5. Vue 使用中的小技巧(山东数漫江湖)

    在vue的使用过程中会遇到各种场景,当普通使用时觉得没什么,但是或许优化一下可以更高效更优美的进行开发.下面有一些我在日常开发的时候用到的小技巧,在下将不定期更新~ 1. 多图表resize事件去中心 ...

  6. Hadoop和大数据:60款顶级开源工具(山东数漫江湖)

    说到处理大数据的工具,普通的开源解决方案(尤其是Apache Hadoop)堪称中流砥柱.弗雷斯特调研公司的分析师Mike Gualtieri最近预测,在接下来几年,“100%的大公司”会采用Hado ...

  7. Hadoop大数据生态系统及常用组件(山东数漫江湖)

    经过多年信息化建设,我们已经进入一个神奇的“大数据”时代,无论是在通讯社交过程中使用的微信.QQ.电话.短信,还是吃喝玩乐时的用到的团购.电商.移动支付,都不断产生海量信息数据,数据和我们的工作生活密 ...

  8. Vue前端开发规范(山东数漫江湖)

    一.强制 1. 组件名为多个单词 组件名应该始终是多个单词的,根组件 App 除外. 正例: export default { name: 'TodoItem', // ... } 反例: expor ...

  9. BTA 常问的 Java基础40道常见面试题及详细答案(山东数漫江湖))

    八种基本数据类型的大小,以及他们的封装类 引用数据类型 Switch能否用string做参数 equals与==的区别 自动装箱,常量池 Object有哪些公用方法 Java的四种引用,强弱软虚,用到 ...

随机推荐

  1. 201621044079 week13 网络

    作业13-网络 1. 本周学习总结 以你喜欢的方式(思维导图.OneNote或其他)归纳总结多网络相关内容. 2. 为你的系统增加网络功能(购物车.图书馆管理.斗地主等)-分组完成 为了让你的系统可以 ...

  2. 关于FEer发展方向的思考

    今天学习了HTTP权威指南这本书,虽然标题是对FEer发展的思考,不过我打算稍后再说这个议题,先对今天学习的内容做个总结. 首先:原来访问服务器的方式有多重,核心是URI,也就是统一资源定位,按照访问 ...

  3. Print之modile, level

    一般print打印的design都会引入module, level. xxxprint(module, level, arg,...)... 每个Module都可以有各自的bitmap,代表这个mod ...

  4. MySQL常用存储引擎功能与用法详解

    本文实例讲述了MySQL常用存储引擎功能与用法. MySQL存储引擎主要有两大类: 1. 事务安全表:InnoDB.BDB. 2. 非事务安全表:MyISAM.MEMORY.MERGE.EXAMPLE ...

  5. 【EF】Entity Framework 6新特性:全局性地自定义Code First约定

    应用场景 场景一:EF Code First默认使用类名作为表名,如果我们需要给表名加个前缀,例如将类名Category映射到表Shop_Category.将Product映射到Shop_Produc ...

  6. Trie字典树的学习及理解

    字典树详解见此 我这里学习时主要是看了李煜东的进阶指南里的讲解,以下是书中介绍的内容. Trie,又称字典树,是一种用于实现字符串快速检索的多叉树结构,Tire的每个节点都拥有若干个字符指针,若在插入 ...

  7. [HZOI2016]偏序&[HZOI2015]偏序II K维偏序问题

    description Cogs: [HZOI2016]偏序 [HZOI2015]偏序 II data range \[ n\le 5\times 10^4\] solution 嵌套\(CDQ\)的 ...

  8. 【以前的空间】BIT的两个小小运用

    剩下一点点时间,就来说说最近才会的关于bit的两个妙用. 求一组数的逆序对 求最长不下降序列 其实两个东西思想差不多,就已第一个为例讲讲. 就是所有数排一遍后,再按照原序列顺序(从后往前),做如下操作 ...

  9. BZOJ1878:[SDOI2009]HH的项链——题解

    http://www.lydsy.com/JudgeOnline/problem.php?id=1878 题面源于洛谷 题目背景 无 题目描述 HH 有一串由各种漂亮的贝壳组成的项链.HH 相信不同的 ...

  10. CC DGCD:Dynamic GCD——题解

    https://vjudge.net/problem/CodeChef-DGCD https://www.codechef.com/problems/DGCD 题目大意: 给一颗带点权的树,两个操作: ...