Activity是Android四大组件之首,其重要性不言而喻,Activity的生命周期更是我们了解Android工作机制的重中之重。我们一般将Activty的生命周期做两种情况下的理解,即正常情况和异常情况。

  所谓正常情况下的生命周期,指的是在有用户参与的情况下,Activity所经过的生命周期的改变;

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

  接下来我们来依次深入了解一下这两种情况下Activity的生命周期。

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

  正常情况下Activity的生命周期如下图所示:

  (1) onCreate() :Activity正在被创建时调用的生命周期函数;

  (2) onStart() :Activity能被用户看到时候调用的生命周期函数;

  (3) onResume() :Activity获取用户焦点的时候调用的生命周期函数;

  (4) onPause() :Activity失去用户焦点的时候调用的生命周期函数;

  (5) onStop() :Activity不能被用户看到的时候调用的生命周期函数;

  (6) onDestroy() :Activity即将被销毁的时候调用的生命周期函数;

  (7) onRestart() :Activity不可见而没有被销毁的情况下重新可见时调用的生命周期函数。

【注意】:

  (1)生命周期函数基本上是配对的,如 onCreate() 方法和 onDestroy() 方法、onStart() 方法和 onStop() 方法、onResume() 方法和 onPause() 方法等;

  (2)当打开一个新的 Activity 时,旧 Activity 的 onPause() 方法会先于新 Activity 的 onResume() 方法执行;

  (3)由于上一条,因此不能在 onPause() 方法中做太过好事的操作,较为耗时的操作应该尽量放在 onStop() 方法中执行,从而使新 Activity 尽快显示出来。

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

  开头说过,导致Activity生命周期异常的情况有两种:Activity的配置发生改变,以及Activity被系统回收。

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

  Activity中有很多配置信息,如果这些信息发生改变,那么Activty就可能会被杀死并重建。但是,我们可以通过代码来干预这些配置,设置某些配置即使发生改变,Activity也不会被杀死。系统配置中有很多内容,如果当某项内容发生改变后,我们不想系统重新创建Activity,则可以在Manifest文件中给这个Activity指定configChanges属性,示例代码如下:

<activity
android:name=".ui.activity.LoginActivity"
android:configChanges="orientation|keyboardHidden|orientation|screenSize"
android:windowSoftInputMode="adjustResize|stateVisible">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>

  例如,我们可以通过设置 android:configChanges 属性来禁止设置在横竖屏切换后被杀死并重建,只需要在Manifest文件的 <activity> 标签中添加 android:configChanges="orientation|keyboardHidden|screenSize" 即可。

  更多的配置选项请参考下表:

  【注意】:不同的配置选项之间可以通过“ | ”符号分隔。

  如果我们不为Activity设置configChanges属性,那么当Activity被杀死重建之后,我们还能不能获取到之前的数据了呢?答案是可以的,我们可以通过Android系统为我们提供的两个方法 onSaveInstanceState() 和 onRestoreInstanceState() 来存储和取出数据。当Activity在异常情况下需要重新创建时,系统会默认为我们保存一些数据,如当前Activity的视图结构、文本框中用户输入的数据、ListView滚动的位置等。  

  如果我们需要将自定义的数据保存下来,可以重写 onSaveInstanceState() 和 onRestoreInstanceState() 这两个方法,使用Bundle对象存储自定义的数据,进行数据传递。

  【注意】:

  (1) onSaveInstanceState() 在onStop()方法之前执行, onRestoreInstanceState() 在onStart()方法之后执行;

  (2) onSaveInstanceState() 和 onRestoreInstanceState() 在正常情况的生命周期中不会被调用,只有当生命周期发生异常时才会被调用;

  (3) onCreate() 方法中有一个Bundle类型的参数,这个参数也是用于接收异常情况下传递的保存数据的,但是不建议在onCreate()方法中处理保存的数据,原因有二:第一,onCreate()方法中处理的事情已经够多了,我们应该将恢复数据的操作提到其他生命周期函数中;第二,每次开启Activity都会调用onCreate()方法,如果在onCreate()方法中恢复数据,那么每次都需要判断该Activity是否被重启了,即Bundle对象是否为空,这样很繁琐;

  (4)和Activity相同,View中也有 onSaveInstanceState() 和 onRestoreInstanceState() 两个方法。Android系统保存和回复View层次结构中的数据的方法使用了委托思想,其流程是这样的:首先Activity被异常终止时,Activity会调用 onSaveInstanceState() 方法去保存数据,然后Activity委托Window去保存数据,接着Window再委托它上面的顶层容器去保存数据,最后顶层容器再去一一通知它的子元素来保存数据。数据恢复过程也是类似的,这样就保证了View层级树中的所有View中的数据都被保存并恢复了。

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

  这里我们需要先了解一下Andorid系统中Activity的优先级分类,从高到低可以分为以下三类:

  (1)前台Activity——正在和用户交互的(获取用户焦点的)Activity,优先级最高;

  (2)可见但非前台Activity——用户可见但没有获取用户焦点的Activity;

  (3)后台Activity——用户不可见、已经被暂停了的的Activity,优先级最低。

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

  【注意】:如果一个进程中没有四大组件在执行,那么这个进程将很快被系统杀死,因此,一些后台工作不适合脱离四大组件而独立的运行在后台中,这样进程将很容易被杀死。比较好的方法是将后台工作放入Service中从而保证进程有一定的优先级,这样就不会轻易地被系统杀死。

【Android - 组件】之Activity生命周期的全面分析的更多相关文章

  1. Android 组件系列-----Activity生命周期

    本篇随笔将会深入学习Activity,包括如何定义多个Activity,并设置为默认的Activity.如何从一个Activity跳转到另一个Activity,还有就是详细分析Activity的生命周 ...

  2. Android-管理Activity生命周期 -停止和重启Activity

    停止和重启activity在activity的生命周期中很重要,它能让用户感觉你的app总是激活的而且不会丢失他们的进度.activity在下面的这些情况会停止和重启: 用户打开常用app窗口然后从你 ...

  3. Android-管理Activity生命周期 -暂停和恢复一个Activity

    在正常的使用app时,前台的activity有时候会被可见的组件阻塞导致activity暂停.比如,当打开一个半透明的activity(就像打开了一个对话框),之前的activity就会暂停.只要ac ...

  4. Android-管理Activity生命周期 -重新创建Activity

    按照正常的app行为,很少情况下activity会销毁,只有当用户点击了返回按钮或者activity通过调用finish()发出销毁信号.系统也有可能销毁activity如果它是停止状态并且很久没有使 ...

  5. Android-管理Activity生命周期 -开始一个Activity

    很多程序都是从main()方法开始启动的,和其他程序不同,android是在activity生命周期的特定状态的特定回调方法中初始化代码的.activity启动和销毁的时候都用很多回调方法. 这里将要 ...

  6. 【Android实验】第一个Android程序与Activity生命周期

    目录 第一个Android程序和Activity生命周期 实验目的 实验要求 实验过程 1. 程序正常启动与关闭 2. 外来电话接入的情况 3. 外来短信接入的情况 4. 程序运行中切换到其他程序(比 ...

  7. Android 四大组件之Activity生命周期

    写这篇博文之前,已经对android有一定的了解和认识.这篇博文主要讲述android的Activity的生命周期,这是android开发者必须掌握的知识.android的Activity组件拥有7个 ...

  8. 深入剖析Android四大组件(一)——Activity生命周期具体解释

    1.管理Activity的生命周期 不管是正在执行的Activity还是没有执行的Activity,它们都接受Android的框架管理,这使得Activity处于不同的生命周期. ①Activity的 ...

  9. Android体系结构及activity生命周期

    Android的系统架构采用了分层架构的思想,如图1所示.从上层到底层共包括四层,分别是应用程序程序层.应用框架层.系统库和Android运行时和Linux内核 Android的系统架构图    每层 ...

随机推荐

  1. .Net Core WebApi(三)在Linux服务器上部署

    鸽了好久,终于有个时间继续写了,继上一篇之后,又写(水)了一篇,有什么不足之处请大家指出,多谢各位了. 下面有两个需要用到的软件,putty和pscp,我已经上传到博客园了,下载请点击这里. 一.准备 ...

  2. yum安装PHP升级到7.1版本

    yum安装PHP升级到7.2版本卸载原来低版本的PHP rpm -qa |grep php|xargs rpm -e 更新yum源 //CentOS/RHEL 7.xrpm -Uvh https:// ...

  3. Mongo 导出为csv文件

    遇到需要从Mongo库导出到csv的情况,特此记录. 先贴上在mongo目录下命令行的语句: ./mongoexport -h 10.175.54.77 -u userName -p password ...

  4. 学习笔记49_Redis

    Redis和memcache区别: 1 . mm是通过客户端驱动实现集群化,Redis是通过服务器配置文件集群 2. redis是可以进行持久化的存储 3. redis提供高级的数据结构,队列,栈都提 ...

  5. 基于node的前端组件包发布至nexus和npmjs

    目录 目录... 3 1. 前言... 1 2. 配置... 1 2.1. 建立组件的导出模块... 1 2.2. 建立组件入口文件... 1 2.3. 配置“ng-package.json”文件.. ...

  6. numpy.array 中的运算

    简单运算 现在有有个需求,给定一个数组,让数组中每一个数乘以2,怎么做呢 n = 10 L = [i for i in range(n)] L # [0, 1, 2, 3, 4, 5, 6, 7, 8 ...

  7. 大数据之路day01_1--Java下载、安装等配置

    从今天开始,我就正式的走上大数据的道路了,如果说我为啥要去学习大数据,可能我的初衷是以后可以接触到人工智能方面的技术,后来在自学的过程中发现,学习人工智能,需要扎实的算法,以及对大量数据的处理,再者, ...

  8. 苹果审核ipv6海外解决思路-About APP Store

    原始简书文章地址(也是我自己的) 首先声明,一我不负责涉及你们内部服务器. 二是好好读文章,别人能过,你们也能过 苹果6月1日出的IPV6协议阻碍了国内大多数积极开发者,我司也不外乎,经过三次被拒后, ...

  9. JS面试题-<变量和类型>-JavaScript浅拷贝与深拷贝

    前言 最开始了解到深浅拷贝是因为准备面试,但那个时候因为在学校做的项目比较少需求也比较简单,所以没有在项目中遇到这类问题,所以对这个问题就属于知道这个知识点,看过相关内容,却没有自己的总结,也没有深入 ...

  10. [ PyQt入门教程 ] PyQt5中数据表格控件QTableWidget使用方法

    如果你想让你开发的PyQt5工具展示的数据显得整齐.美观.好看,显得符合你的气质,可以考虑使用QTableWidget控件.之前一直使用的是textBrowser文本框控件,数据展示还是不太美观.其中 ...