设备配置(device configuration)用以描述设备当前状态,包括:屏幕方向、屏幕密度、屏幕尺寸、键盘类型、语言等。配置若在运行时发生变化(runtime configuration change),Android 会寻找更合适的资源以匹配设备配置。
好比旋转设备会改变配置,那么 activity 实例会被系统销毁,而后建立一个新的 activity 实例。 因此数据就丢掉了。html
试想你在用手机看电子书,正读第21页呢,转了屏幕,竟从第1页开始了,会不会很恼火?会。
所以须要采起某种方式保存数据,以便在旋转后恢复到旋转前的交互状态。Android 考虑到了这种状况,提供了一个方法 onSaveInstanceState()
,Android 会在杀掉 activity 前调用它,给了程序员一个机会,像这样;java
private static final String KEY_INDEX = "index"; private int mCurrentIndex = 0; @Override protected void onCreate(Bundle savedInstanceState) { if (savedInstanceState != null) { mCurrentIndex = savedInstanceState.getInt(KEY_INDEX, 0); } } @Override protected void onSaveInstanceState(Bundle outState) { super.onSaveInstanceState(outState); outState.putInt(KEY_INDEX, mCurrentIndex); }
要特别强调的是,onSaveInstanceState 并非生命周期方法,Android 系统并不保证在杀死 activity 前必定调用它:
当 activity 进入后台或者被销毁时 onPause()
总会被调到;当 activity 被销毁时 onStop()
总会被调到;而 onSaveInstanceState()
不会被调到的两种状况:android
当 activity B 被调到 activity A 上面,而且在B的生命周期内A未被杀掉,系统不会为A调用这个方法。由于A仍保持它原来的用户交互的状态。git
当用户按返回键从 activity B 返回到 activity A,系统不会为B调用这个方法。由于B的实例已被销毁,因此没有必要恢复状态。程序员
而严谨的你产生了疑问,既然onPause总会被调到,为何不在onPause中保存数据,而要额外增长一个API作这件事情呢?
答案:要分清场合。
程序员确实须要在onPause中保存数据(在onStop中作可能就来不及了),可是通常状况下保存的是永久性数据,好比preference、database、json file,可能还要给 thread、service 擦屁股;
而一些暂时的数据呢?好比上面讲的状况,只是旋转了屏幕,程序员就能够把当前页数放进 Bundle 中,交给 Application 保管。github
旋转发生了,activity 实例被销毁: onPause() // 存储永久数据 onSaveInstanceState(Bundle) // 存储暂时数据,在onStop前被调用 onStop() onDestroy() 重建 activity 实例: onCreate(Bundle) // 获取暂存数据 onStart() onRestoreInstanceState(Bundle) // 也可在这获取暂存数据,在onStart后被调用 onResume() // 作对应于onPause的恢复的工做
此节内容参考:编程
部份内容翻译自 andriod.app.Activity.onSaveInstanceState()源码注释
部份内容整理自《Android权威编程指南3.2, 3.3》
QMLN31821007的CSDN专栏:Android中onSaveInstanceState(Bundle outState)与OnPause()的区别json
如上述,当设备旋转时,Activity实例将被销毁,那么附加在activity上的fragment也将被销毁。若是你在应用开发中喜欢使用 fragment 管理界面,碰巧某个案例中使用了Fragment管理音频播放,那么在旋转屏幕的时候,不中断音频的播放,体验是否是很棒?是。segmentfault
虽然程序员能够覆写onSaveInstanceState()
用来保存当前播放进度,可是播放会中断。怎么达成这个目的呢?
在Fragment的onCreate()
方法中调用setRetainInstance(true)
.
当旋转发生时,fragment实例会被保留(全部实例变量的值都保持不变),而后传递给新的activity,新的fragment manager依据保留的fragment重建它的视图。数组
对比上述看电子书的情形,假如电子书也是由fragment构建,采用setRetainInstance(true)的方式,也能够保证在旋转时当前页数同fragment一块儿被保留下来。
可是,被保留的fragment有可能会因系统回收内存而被销毁,那么页数也就丢失了。可是采用 onSaveInstanceState(Bundle outState)
的优点在于,bundle对象会保存的久一些(具体多久不清楚 TODO)。
若是Fragment只是简单的UI View,像是TextView, Button, CheckBox, ImageView... 不建议使用setRetainInstance方法,只须要记住当前fragment的index,而后在设备旋转后根据数据从新实例化一个fragment。由于不包含大量数据,旋转的过程当中几乎能够用“无缝切换”来形容。
一个相关问题
上述的讨论全是基于“设备旋转”的状况下,如何暂时地保存数据;若是Activity进入后台,程序员仍是须要在onPause中长久地保存播放/阅读进度的。
此节内容参考:
内容整理自《Android权威编程指南14》
旋转设备屏幕时WebView将从新加载网页,这是由于WebView包含了太多的数据,以致没法在onSaveInstanceState(...) 方法内保存全部数据。
对于一些相似的类(如VideoView), Android文档推荐让activity本身处理设备配置变动。也就是说,无需销毁重建activity可直接调整本身的视图以适应新的屏幕尺寸。这样, WebView也就没必要从新加载所有数据了。不适用于全部视图(TODO不清楚哪些视图)。
<!-- Activity将本身处理设备配置的改变(键盘隐藏、屏幕方向改变、屏幕大小改)> <activity android:name=".PhotoPageActivity" android:configChanges="keyboardHidden|orientation|screenSize" />
此节内容参考:
内容整理自《Android权威编程指南31.3.2》
继承View以实现自定义View的绘制,好比矩形框
设备旋转后,绘制的矩形框会消失,要解决这个问题,可使用如下View方法:
protected Parcelable onSaveInstanceState(); protected void onRestoreInstanceState(Parcelable state);
Parcelable onSaveInstanceState()
建议保存的数据不该该是持久的或者稍后难以重建的。好比不该该保存屏幕的当前位置,由于位置会被从新计算。应该保存的,好比list view当前选中的条目。
与Activity和Fragment对应方法的区别是,代替Bundle参数,View的方法返回并处理的是实现了Parcelable接口的对象实例。
Parcelable接口容许类的实例能够被写入Parcel,并从中恢复,[官方文档示范了如何Parcelable化一个对象](
http://developer.android.com/intl/zh-cn/...。
能够把全部的元数据类型(Primitives)(byte, double, float, int, long, String)写入Parcel;
也能够把元数据的数组(Primitive Arrays)写入Parcel;
能够把Parcelable接口的对象写入Parcel;
能够把Bundle写入Parcel;
能够把Untyped Containers(Object, Object[], List)写入Parcel。
在自定义的View BoxDrawingView
内定义一个继承自BaseSavedState
的静态类BoxDrawingState
,实现writeToParcel(Parcel out, int flags)
方法,参考github commit
public class BoxDrawingView extends View { private ArrayList<Box> mBoxes = new ArrayList<Box>(); static class BoxDrawingState extends BaseSavedState { ArrayList<Box> boxes; private BoxDrawingState(Parcel in) { super(in); in.readList(boxes, null); } @Override public void writeToParcel(Parcel out, int flags) { super.writeToParcel(out, flags); out.writeList(boxes); } } @Override protected Parcelable onSaveInstanceState() { Parcelable superState = super.onSaveInstanceState(); BoxDrawingState boxDrawingState = new BoxDrawingState(superState); boxDrawingState.boxes = mBoxes; return boxDrawingState; } @Override protected void onRestoreInstanceState(Parcelable state) { BoxDrawingState boxDrawingState = (BoxDrawingState) state; super.onRestoreInstanceState(boxDrawingState.getSuperState()); mBoxes = boxDrawingState.boxes; } }
另外一种暂存ArrayList of Custom Objects的方法,Parcelable化自定义类,调用Bundle.putParcelableArrayList(String key, ArrayList<? extends Parcelable> value)
. 参考 github commit
此节内容参考:
内容整理自《Android权威编程指南32.5》
How to prevent custom views from losing state across screen orientation changes-方法1
How to prevent custom views from losing state across screen orientation change-方法2
Android View onSaveInstanceState not called
版权声明:《如何在Android设备旋转时暂存数据以保护当前的交互状态?》由 WeiYi.Li 在 2015年11月08日写做。著做权归做者全部。商业转载请联系做者得到受权,非商业转载请注明出处。
文章连接:http://li2.me/2015/11/handling-android-r...