一、Activity 的形态
Active/Running:
Activity处于活动状态,此时Activity处于栈顶,是可见状态,可与用户进行交互。
Paused:
当Activity失去焦点时,或被一个新的非全屏的Activity,或被一个透明的Activity放置在栈顶时,Activity就转化为Paused状态。但我们需要明白,此时Activity只是失去了与用户交互的能力,其所有的状态信息及其成员变量都还存在,只有在系统内存紧张的情况下,才有可能被系统回收掉。
Stopped:
当一个Activity被另一个Activity完全覆盖时,被覆盖的Activity就会进入Stopped状态,此时它不再可见,但是跟Paused状态一样保持着其所有状态信息及其成员变量。
Killed:
当Activity被系统回收掉时,Activity就处于Killed状态。 Activity会在以上四种形态中相互切换,至于如何切换,这因用户的操作不同而异。了解了Activity的4种形态后,我们就来聊聊Activity的生命周期。
二、Activity 生命周期一览
三、验证生命周期调用
-
1.我们先来分析Activity启动过程中所调用的生命周期方法,运行程序截图如下:
从Log中我们可以看出Activity启动后,先调用了onCreate方法,然后是onStart方法,最后是onResume方法,进入运行状态,此时Activity已在前台显示。因此, Activity启动–>onCreate()–>onStart()–>onResume()依次被调用。
-
2.当前Activity创建完成后,按Home键回到主屏。按如上操作运行截图:
我们在Activity创建完成后,点击Home回调主界面时,可以发现此时onPause方法和onStop方法被执行,也就是点击Home键回到主界面(Activity不可见)–>onPause()–>onStop()依次被调用。
-
3.当我们点击Home键回到主界面后,再次点击App回到Activity时,调用结果如下:
我们可以发现重新回到Activity时,调用了onRestart方法,onStart方法,onResume方法。因此,当我们再次回到原Activity时–>onRestart()–>onStart()–>onResume()依次被调用。
-
4.当我们在原有的Activity的基础上开启新的Activity时,调用结果如下:
调用了 A 的onPause方法、onStop方法,B 的onCreate方法、onStart方法、onResume方法。因此当我们开启新的Activity时,A.onPause()->B.onCreate()->B.onStart()->B.onResume()->A.onStop()依次被调用。注意:如果新的ActivityB使用了透明主题,原ActivityA的onStop()方法不会调用。
同时,新的ActivityB生命周期方法是在原ActivityA的onPause方法执行后调用,所以onPause方法不能操作耗时任务。特例:一个前台activity启动模式为singleTop,再次启动当前activity的生命周期如下:onPause()->onNewIntent()->onResume()。
-
5 当我们点击Back键回退时,回调结果如下:
从Log我们可以看出,当点击Back键回退时,调用了 A 的onRestart方法、onStart方法、onResume方法,B 的onPause方法、onStop方法、onDestroy方法。因此当我们点击Back键回退时,B.onPause()->A.onRestart()->A.onStart()->A.onResume()-> B.onStop()->B.onDestroy()依次被调用。
四、异常情况下的Activity生命周期
异常的生命周期主要分以下两种情况:系统内存不足Activity被系统回收,当前设备的Configuration发生变化(一般指横竖屏切换)从而导致Activity被销毁重建。
1、相关的系统配置发生改变导致Activity被杀死并重新创建(一般指横竖屏切换)
从Log中我们可以看出当我们正常启动Activity时,onCreate,onStart,onResume方法都会依次被回调。而如果我们此时把竖屏的Activity人为的调整为横屏,我们可以发现onPause,onSaveInstanceState,onStop,onDestroy,onCreate,onStart,onRestoreInstanceState,onResume依次被调用,单从调用的方法我们就可以知道,Activity先被销毁后再重新创建。
当系统配置发生变化后,Activity会被销毁,会自动调用onSaveInstanceState方法来保存当前Activity的状态信息,因此我们可以在onSaveInstanceState方法中存储一些数据以便Activity重建之后可以恢复这些数据。当Activity被重新创建之后,系统还会去调用onRestoreInstanceState方法,并把Activity销毁时通过onSaveInstanceState方法保存的Bundle对象作为参数同时传递给onRestoreInstanceState和onCreate方法,因此我们可以通过onRestoreInstanceState和onCreate方法来判断Activity是否被重新创建,倘若被重建了,我们就可以对之前的数据进行恢复。这里有点需要特别注意,onSaveInstanceState和onRestoreInstanceState只有在Activity异常终止时才会被调用的,正常情况是不会调用这两个方法的。
到这里大家可能还有一个疑问,onRestoreInstanceState和onCreate方法都可以进行数据恢复,那到底用哪个啊?其实两者都可以,两者的区别在于,onRestoreInstanceState方法一旦被系统回调,其参数Bundle一定不为空,无需额外的判断,而onCreate的Bundle却不一定有值,因为如果Activity是正常启动的话,Bundle参数是不会有值的,因此我们需要额外的判断条件,当然虽说两者都可以数据恢复,但更倾向于onRestoreInstanceState方法。
最后还有点我们要知道的是,在onSaveInstanceState方法和onRestoreInstanceState方法中,android系统会自动帮我们恢复一定的数据,如当前Activity的视图结构,文本框的数据,ListView的滚动位置等,这些View相关的状态系统都会帮我们恢复,这是因为每个View也有onSaveInstanceState方法和onRestoreInstanceState方法。
2、系统内存不足导致低优先级的Activity被杀死
当系统内存不足的时候,系统就会按照一定的优先级去杀死目标Acitivity的进程来回收内存,并且此时Activity的onSaveInstanceState方法会被调用来存储数据,并在后续Activity恢复时调用onRestoreInstanceState方法来恢复数据,其数据的存储过程和恢复过程跟上面的情况差不多。
Android 进程层次
在android系统中最重要的进程被称为前台进程,然后依次是任何可见进程、服务进程、后台进程,最后是空进程。接下来我们将进一步展开。
在开始之前我们先要明确一个问题,当我们谈论进程优先级的时候是以 activity、service 这样的组件来说的,但请注意这些优先级是在进程的级别上的,而非组件级别。只要有一个组件是前台进程,就会将整个进程变为前台进程。同时我们要知道绝大多数应用是单进程的,如果我们有生命周期差异很大的不同部分或者某个部分非常重量型,那么我们强烈建议把它们分为不同的进程,以便让重量级进程尽早被系统回收。
明白这点后我们再看看下面一张图(图来自Who lives and who dies? Process priorities on Android):
从图中我们可以看出共有5中优先级线程,Foreground Processes ,Visible Processes ,Service Processes , Background Processes , Empty Processes ;
- 1.Foreground Processes(前台进程)
系统中前台进程的数量很少(这点从图中也是可以看出来的), 前台进程几乎不会被杀死. 只有当内存低到无法保证所有的前台进程同时运行时才会选择杀死某个前台进程.以下几种都属于前台进程:
a. 处于前台正与用户交互的activity
b. 与前台activity绑定的service
c. 调用了startForeground()方法的service
d. 正在执行onCreate(), onStart(), 或onDestroy()方法的service
e. 正在执行onReceive()方法的BroadcastReceiver.
凡是包含以上任意一种情况的进程都是前台进程。当然一些 activity 在依靠他们成为前台进程的同事,也可能依赖 bound service 。任何进程,如果它持有一个绑定到前台 activity 的服务,那么它也被赋予了同样的前台优先级。
- 2.Visible Processes(可视进程)
此时如果一个Activity可见但并非处于前台时,如在Activity中弹出了一个对话框,从而导致Activity可见但位于后台无法与用户交互,这个进程就可以被视为可见进程,同时我们也必须明白可见 activity 的 bound service 和 content provider 也处于可见进程状态。这同样是为了保证使用中的 activity 所依赖的进程不会被过早地杀掉。但还是需要注意的是,只是可见并不意味着不能被杀掉。如果来自前台进程的内存压力过大,可见进程仍有可能被杀掉。从用户的角度看,这意味着当前 activity 背后的可见 activity 会被黑屏代替。当然,倘若我们正确地重建 activity ,在前台 activity 关闭之后,我们的进程和 activity 会立刻恢复而没有数据损失。
- 3.Service Processes(服务进程)
如果我们通过startService()启动一个service服务,那么它被看作是一个服务进程。对于许多在后台做处理(如异步加载数据,获取耗时资源等)而没有立即成为前台服务的app都属于这种情况。这是比较常见也是最合理的后台处理方式,这种进程只有在可见进程和前台进程内存不足时才有可能被杀掉。
- 4.Background Processes(后台进程)
假如我们的Activity目前是前台进程,但是这时候,我们点Home键,将导致onPause,onStop方法被调用,我们的进程也就变成了后台进程,当然我们的后台进程并不会被立马杀死,所以这些进程会保留一段时间,直到更高优先级进程需要内存的时候才被回收,并且是按照最近最少使用顺序(最少使用的会被优先回收)。很多时候我们会发现手机的内存大都是被后台App进程占用了大部分空间,而android系统这样做的好处是可以使用我们在下次重新打开此进程的app时无需重新分配和加载资源,从而拥有更好的用户体验。
。然而内存不足的时候,他们仍然会被杀掉,所以我们应该和可见 activity 处理情况一样,应该尽量能够在不丢失用户状态的情况下重建这些 activity ,以便达到更佳的用户体验。
- 5.Empty Processes(空进程)
在任何层次中,空进程都是最低优先级的。如果我们的进程不属于以上类别,那它就是空进程。空进程是没有活跃的组件,只是出于缓存的目的而被保留(为了更加有效地使用内存而不是完全释放掉),只要 Android 系统内存需要可以随时杀掉它们。
嗯,现在我们应该对android进程有个比较明确的概念了,回到之前的的Activity内存不足被android系统杀死的话题,为了防止一些重要的Activity不被意外杀死,我们应该让当前所在进程的Activity保持较高的优先级,如使其变为前台进程,或者通过service去绑定,也可以让其成为单独的进程。当然如果真的不幸被杀死,我们也应该尽量通过onSaveInstanceState方法和onRestoreInstanceState方法来保存和恢复数据,以便获得更佳的用户体验。
解决Activity销毁重建问题
通过上面的分析我们知道当系统配置发生变化后,Activity会被重建,那有没有办法使其不重建呢?方法自然是有的,那就是我们可以给Activity指定configChange属性,当我们不想Activity在屏幕旋转后导致销毁重建时,可以设置configChange=“orientation”;当SDK版本大于13时,我们还需额外添加一个“screenSize”的值,对于这两个值含义如下:
orientation:屏幕方向发生变化,配置该参数可以解决横竖屏切换时,Activity重建问题(API<13)
screenSize:当设备旋转时,屏幕尺寸发生变化,API>13后必须配置该参数才可以保证横竖切换不会导致Activity重建。
说白了就是设置了这两个参数后,当横竖屏切换时,Activity不会再重建并且也不会调用之前相关的方法,取而代之的是回调onConfigurationChanged方法。案例代码如下:
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.cmcm.activitylifecycle" >
<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:supportsRtl="true"
android:theme="@style/AppTheme" >
<activity android:name=".MainActivity"
android:configChanges="orientation|screenSize" >
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<activity android:name=".SecondActivity"></activity>
</application>
</manifest>
在MainActivity中我们重写onConfigurationChanged,原型如下:
@Override public void onConfigurationChanged(Configuration newConfig) {
super.onConfigurationChanged(newConfig);
LogUtils.e("onConfigurationChanged is invoke!!!");
}
我们运行程序,然后执行横竖屏切换,看看打印的Log信息:
从Log可以看到Activity确实没有重建,并且也没有回调onSaveIntanceState方法和RestoreInstanceState方法来存储或者恢复数据,相反的是onConfigurationChanged方法被回调了,因此我们在这种情况下,可以在此方法中做一些额外的工作。
五、总结
Activity的主要生命周期方法有onCreate()、onStart()、onRestart()、onResume()、onPause()、onStop()、onDestroy()。
1、Activity转为后台时调用onPause(),不可见时调用onStop();反之转为前台时调用onResume(),可见时调用onRestart()、onStart()。
2、页面切换时先调用原Activity的onPause(),再调用新Activity的onCreate()、onStart()、onResume(),然后再调用原Activity的onStop()。所以避免在onPause()做耗时操作。
3、异常情况下(包括横竖屏切换和系统内存不足时Activity被杀死),系统都会自动调用onSaveInstanceState和onRestoreInstanceState方法,来存储和恢复数据。调用时机:onPause()后调用onSaveInstanceState(),onStart()后调用onRestoreInstanceState()。
4、要避免横竖屏切换时Activity的销毁重建,可以在AndroidManifest中指定Activity的configChange属性为"orientation|screenSize"。当横竖屏切换时,Activity不会再重建并且也不会调用之前相关的方法,取而代之的是回调onConfigurationChanged方法。
参考文章:
1、Android之Activity生命周期浅析