你真的了解Activity的生命周期?
上面这张图所有的Android开发者想必都非常熟悉了,但是这个图上隐藏的一些重要的信息点有很多被我们忽视了。下面我们就一起来梳理一下。
在《Android开发艺术探索中》一书中,对Activity的生命周期分析分为两部分内容:
一部分是典型情况下的生命周期,另一方面是异常情况下的生命周期。
本文就先对书上的内容要点进行一些总结。
典型情况下的生命周期分析
在正常情况下,Activity会经历如下生命周期。
1.onCreate 表示Activity正在被创建。
2.onRestart 表示Activity正在重新启动,这种情形一般是用户行为导致,比如用户按HOME键切换到桌面,此时当前的Activity就会暂停,onPause和onStop被执行,接着用户重新回到这个Activity,就会出现onRestart。
3.onStart 表示Activity正在被启动,即将开始,这时Acivity已经被设置为可见了,但还没有出现在前台,不能与用户交互。
4.onResume 表示Activityi经可见了,并出现在前台开始活动。
5.onPause 表示Activity正在停止。前一个Activity的onPause必须先执行完,新的Activity的onResume才会执行。
6.onStop 表示Activity即将停止,在这里可以稍微做一些重量级的回收工作,同样不能太耗时。
7.onDestroy 表示Activity即将被销毁,在这里可以做一些回收工作和资源释放。
关键点
1.打开新的的Activity时,如果新的Activity采用了透明主题,那么当前Activity不会回调onStop。
2.onStart和onStop是从Activity是否可见这个角度来回调的,而onResume和onPause是从Activity是否位于前台的角度来回调的。
3.不能在onPause中做重量级操作,因为必须onPause执行完成后,新的Activity才能Resume。
异常情况下的生命周期分析
在一些异常情况下,比如当资源相关的系统配置发生改变以及系统内存不足时,Activity就可能被杀死。
实例状态 InstanceState
异常状态导致Activity被杀死并重新创建:
在Activity被销毁时,他的onPause、onStop、onDestroy均会被调用,同时由于Activity是在异常情况下终止的,系统会调用onSaveInstanceState来保存当前Activity的状态。这个方法的调用时机是在onStop之前。
在Activity被重新创建后,系统会调用onRestoreInstanceState,并把Activity销毁时保存的Bundle对象作为参数传给onRestoreInstanceState和onCreate方法。这个方法的调用时机时在onStart之后。
关键点
- 系统只有在Activity异常终止的情况下才会调用onSaveInstanceState和onRestoreInstanceState来保存和恢复实例状态。
- 在onCreate里使用savedInstance时千万要注意判空
- 资源内存不足导致Activity被杀死情况,系统会按照进程优先级的顺序杀死目标Activity所在的进程。
Android的进程优先级
android将进程的优先级分为5个层次,按照优先级由高到低排列如下:
前台进程(Foreground process)。它表明用户正在与该进程进行交互操作,android系统依据下面的条件来将一个进程标记为前台进程:
*该进程持有一个用户正在与其交互的Activity(也就是这个activity的生命周期方法走到了onResume()方法)。
*该进程持有一个Service,并且这个Service与一个用户正在交互中的Activity进行绑定。
*该进程持有一个前台运行模式的Service(也就是这个Service调用了startForegroud()方法)。
*该进程持有一个正在执行生命周期方法(onCreate()、onStart()、onDestroy()等)的Service。
*该进程持有一个正在执行onReceive()方法的BroadcastReceiver。
一般情况下,不会有太多的前台进程。杀死前台进程是操作系统最后无可奈何的做法。当内存严重不足的时候,前台进程一样会被杀死。
可见进程(Visible process)。它表明虽然该进程没有持有任何前台组件,但是它还是能够影响到用户看得到的界面。android系统依据下面的条件将一个进程标记为可见进程:
*该进程持有一个非前台Activity,但这个Activity依然能被用户看到(也就是这个Activity调用了onPause()方法)。例如,当一个activity启动了一个对话框,这个activity就被对话框挡在后面。
*该进程持有一个与可见(或者前台)Activity绑定的Service。
服务进程(Service process)。除了符合前台进程和可见进程条件的Service,其它的Service都会被归类为服务进程。
后台进程(Background process)。持有不可见Activity(调用了onStop()方法)的进程即为后台进程。通常情况下都会有很多后台进程,当内存不足的时候,在所有的后台进程里面,会按照LRU(最近使用)规则,优先回收最长时间没有使用过的进程。
空进程(Empty process)。不持有任何活动组件的进程。保持这种进程只有一个目的,就是为了缓存,以便下一次启动该进程中的组件时能够更快响应。当资源紧张的时候,系统会平衡进程缓存和底层的内核缓存情况进行回收。
如果一个进程同时满足上述5种优先级中的多个等级条件,android系统会优先选取其中最高的等级作为该进程的优先级。例如,一个进程持有一个Service(服务进程等级)和一个前台Activity(前台进程等级),那么操作系统会将这个进程标记为前台进程。
另外需要注意的是,如果一个进程为另外一个进程提供服务,那么这个进程的优先级不会低于享受服务的进程。例如,假设进程A中的Content Provider为进程B提供服务,或者进程A中有一个Service与进程B中的组件进程绑定,那么进程A的优先级至少要与进程B一致,或者高于进程B。
参考文献:
《Android开发艺术探索》
林佳楠 Android的进程优先级