屏幕旋转时 Activity 的生命周期 —— 测试与结论

时间:2024-11-14 00:06:08

关于 Android 手机横竖屏切换时 Activity 的生命周期问题,网上有很多相似的文章,大多数都是说明在竖屏切换横屏时 Activity 会重启一次,而在横屏切换竖屏时 Activity 会重启两次。

我本身不太理解这样设计的意义,并且觉得新版本会解决这个问题,所以做了一项测试,测试环境为 Android 7.0 模拟器。

测试

测试首先创建一个新的 Activity,并在其各个生命周期方法内打印日志,并执行如下操作得到结果:

在 targetSdkVersion 的值大于 12 时

  1. Activity启动
I/System.out: onCreate
I/System.out: onStart
I/System.out: onResume
  1. 竖屏切换为横屏
I/System.out: onConfigurationChanged
I/System.out: onPause
I/System.out: onSaveInstanceState
I/System.out: onStop
I/System.out: onDestroy
I/System.out: onCreate
I/System.out: onStart
I/System.out: onRestoreInstanceState
I/System.out: onResume
  1. 横屏切回竖屏
I/System.out: onConfigurationChanged
I/System.out: onPause
I/System.out: onSaveInstanceState
I/System.out: onStop
I/System.out: onDestroy
I/System.out: onCreate
I/System.out: onStart
I/System.out: onRestoreInstanceState
I/System.out: onResume
  1. 配置 android:configChanges="orientation|screenSize" 后,竖屏切换为横屏
I/System.out: onConfigurationChanged
  1. 配置 android:configChanges="orientation|screenSize" 后,横屏切回竖屏
I/System.out: onConfigurationChanged

在 targetSdkVersion 的值小于或等于 12 时

  1. Activity启动
I/System.out: onCreate
I/System.out: onStart
I/System.out: onResume
  1. 竖屏切换为横屏
I/System.out: onConfigurationChanged
I/System.out: onPause
I/System.out: onSaveInstanceState
I/System.out: onStop
I/System.out: onDestroy
I/System.out: onCreate
I/System.out: onStart
I/System.out: onRestoreInstanceState
I/System.out: onResume
  1. 横屏切回竖屏
I/System.out: onConfigurationChanged
I/System.out: onPause
I/System.out: onSaveInstanceState
I/System.out: onStop
I/System.out: onDestroy
I/System.out: onCreate
I/System.out: onStart
I/System.out: onRestoreInstanceState
I/System.out: onResume
  1. 配置 android:configChanges="orientation" 后,竖屏切换为横屏
I/System.out: onConfigurationChanged
  1. 配置 android:configChanges="orientation" 后,横屏切回竖屏
I/System.out: onConfigurationChanged

总结

可以发现新版本 Android 设备上的运行结果和网上的结论并不完全相同,在横竖屏切换时,都会调用一次 onConfigurationChanged,并重启 Activity 一次,行为完全一致,具体的生命周期为:

I/System.out: onConfigurationChanged
I/System.out: onPause
I/System.out: onSaveInstanceState
I/System.out: onStop
I/System.out: onDestroy
I/System.out: onCreate
I/System.out: onStart
I/System.out: onRestoreInstanceState
I/System.out: onResume

在 targetSdkVersion 的值大于 12 时,配置 android:configChanges="orientation|screenSize"

在 targetSdkVersion 的值小于或等于 12 时,配置 android:configChanges="orientation"

可以做到拦截 Activity 的重新创建:

I/System.out: onConfigurationChanged

后续

将会对过去版本的 Android 设备做相应的测试,来验证得到结果。

改进

通常研究设备旋转带来的 Activity 重启问题,是为了解决 Activity 重启导致成员变量未保存引发运行出错。但之所以 Activity 会重启是因为运行时配置发生了更变,可能有更适合的资源来匹配新的设备配置,例如在横屏时,有更好利用空间的横版布局资源,这时就需要 Activity 重新加载布局显示。

所以,好的解决办法是利用 onSaveInstanceState(Bundle outState) 方法保存成员变量的值到 Bundle 对象中,在 Activity 重启后,再在 onCreate(Bundle savedInstanceState) 中获取这些保存的信息。