Ian Darwin2.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项目。
相关资源:《Android应用开发揭秘》源码