2.4 在用户旋转设备时保存数据
Ian Darwin
2.4.1 问题
当用户旋转设备时,Android通常销毁并重新创建当前活动。你希望在这一周期中保留某些数据,但是在此期间活动中的所有字段都将丢失。
2.4.2 解决方案
对此有多种解决方案。如果所有数据都由简单类型组成(包括String),或者是Serializable类型,就可以在传入的Bundle的onSaveInstanceState()中保存数据。
另一种解决方案是在活动中返回一个任意的对象,实现onRetainNonConfigurationInstance()保存某些值;在onCreate()接近结束的地方调用getLastNonConfigurationInstance()查看是否有过去保存的值,如果有,相应地为字段赋值。
2.4.3 讨论
使用onSaveInstanceState()
参见攻略1.6。
使用onRetainNonConfigurationInstance()
getLastNonConfigurationInstance()方法的返回类型是Object,所以可以返回任何值。你可能希望创建一个Map或者编写内部类来存储这些值,但是传递对当前活动的引用往往更容易,例如,使用this:
/**
*返回在整个应用推倒重建的过程中持续的任意单个令牌对象
*/
@Override
public Object onRetainNonConfigurationInstance() {
return this;
}
上述的方法将在Android销毁主活动时调用。假定你希望保留对运行中的服务更新的另一个对象的引用,该对象由活动中的一个字段引用,可能还有一个表示服务是否活动的布尔值。在上述代码中,返回对活动的引用,从这个引用中可以访问活动的所有字段(当然包括私有字段,因为输入的Activity和输出的Activity对象是同一个类)。以我的地理跟踪应用程序JPSTrack为例,FileSaver类从位置服务接收数据;我希望不管屏幕是否旋转,程序都能连续获取位置并将其存储到磁盘,而不是在每次屏幕旋转的时候都必须重新启动。如果你的设备固定在汽车的仪表盘上(希望如此),旋转就不太可能发生,但是如果旅客或者行人在地理位置跟踪的同时拍摄照片或者作笔记,就很可能发生屏幕旋转。
Android创建新实例之后,调用onCreate()通知新实例已经创建。在onCreate()中,一般进行类构造器操作,例如,初始化字段和指派事件监听器。你仍然需要做这些事情,但我们先把它放在一边。但是,在接近onCreate()结束的地方,将添加一些代码来获得旧的实例(如果有的话),并且从中得到某些重要的字段,代码如例2-3所示。
例2-3:onCreate方法
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
saving = false;
paused = false;
//许多其他的初始化
//现在看看我们是否被旋转等动作中断
Main old = (Main) getLastNonConfigurationInstance();
if (old != null) {
saving = old.saving;
paused = old.paused;
//这是最重要的一行:保存到同一个文件!
fileSaver = old.fileSaver;
if (saving) {
fileNameLabel.setText(fileSaver.getFileName());
}
return;
}
// I/O Helper
fileSaver = new GPSFileSaver(...);
}
fileSaver对象很重要,我们希望它保持运行而不是每次都重新创建。如果没有旧的实例,那么只在onCreate()的最后创建fileSaver;否则就会创建一个新的实例代替旧实例,这至少对于性能来说很不利。
当onCreate()方法结束时,没有对旧实例的任何引用,因此它将会成为Java GC的候选者。
最终的结果是,活动在屏幕旋转期间很好地保持运行,而不是重新创建。
在AndroidManifest.xml中设置android:configChanges="orientation"是可能的替代方案,但是风险较大。
2.4.4 参阅
攻略2.3
2.4.5 源代码下载URL
可以从http://projects.darwinsys.com/jpstrack.android
下载这个例子的源代码。注意,还必须从同一个位置下载jpstrack项目。