我有List<CustomObject>
(其中CustomObject
来自外部库 - 我无法对其进行更改)。我想在onSaveInstanceState(Bundle)
保存这个,但我似乎无法做到。以下是我尝试过的选项:
outState.putSerializable(KEY, (ArrayList<CustomObject>) myList); // because myList is instantiated as an ArrayList
outState.putSerializable(KEY, myList.toArray());
在手机上切换方向时,两个选项工作(是的,切换方向时调用onSaveInstanceState
- 我在logcat中检查过)。但是,当当前活动尝试启动另一个活动(使用startActivity(Intent)
)时,Android会暂停当前活动并再次调用onSaveInstanceState()
。这一次,它失败了,由于某些原因我不知道。可疑的是onSaveInstanceState()
成功执行。打印的堆栈跟踪不指向我的任何代码:
E/AndroidRuntime(23898): java.lang.RuntimeException: Parcel: unable to marshal value my.custom.Object@5e07e43b
E/AndroidRuntime(23898): at android.os.Parcel.writeValue(Parcel.java:1087)
E/AndroidRuntime(23898): at android.os.Parcel.writeArray(Parcel.java:519)
E/AndroidRuntime(23898): at android.os.Parcel.writeValue(Parcel.java:1072)
E/AndroidRuntime(23898): at android.os.Parcel.writeMapInternal(Parcel.java:469)
E/AndroidRuntime(23898): at android.os.Bundle.writeToParcel(Bundle.java:1445)
E/AndroidRuntime(23898): at android.os.Parcel.writeBundle(Parcel.java:483)
E/AndroidRuntime(23898): at android.app.ActivityManagerProxy.activityPaused(ActivityManagerNative.java:1427)
E/AndroidRuntime(23898): at android.app.ActivityThread.handlePauseActivity(ActivityThread.java:3106)
E/AndroidRuntime(23898): at android.app.ActivityThread.access$2400(ActivityThread.java:119)
E/AndroidRuntime(23898): at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1870)
E/AndroidRuntime(23898): at android.os.Handler.dispatchMessage(Handler.java:99)
E/AndroidRuntime(23898): at android.os.Looper.loop(Looper.java:123)
E/AndroidRuntime(23898): at android.app.ActivityThread.main(ActivityThread.java:4363)
E/AndroidRuntime(23898): at java.lang.reflect.Method.invokeNative(Native Method)
E/AndroidRuntime(23898): at java.lang.reflect.Method.invoke(Method.java:521)
E/AndroidRuntime(23898): at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:860)
E/AndroidRuntime(23898): at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:618)
E/AndroidRuntime(23898): at dalvik.system.NativeStart.main(Native Method)
有没有办法在实例状态中存储自定义对象?
答案 0 :(得分:6)
制作CustomObject
工具Parcelable
并使用:
outState.putParcelable(KEY, myList);
onSaveInstanceState(outState);
同时检查this教程。
CommonsWare评论后编辑:
如果您的CustomObject
未实现Serializable
或Parcelable
,我会尝试将其包装在您自己的对象中并添加:
private void readObject(ObjectInputStream aStream) throws IOException, ClassNotFoundException { /*Your deserialization */ }
private void writeObject(ObjectOutputStream aStream) throws IOException { /*Your serialization */}
答案 1 :(得分:5)
让您的List<CustomObject>
由服务持有,并通过本地绑定模式使您的活动可以访问。
您不仅不必担心在实例状态下保留它,而且您可以更好地控制内存中这些对象的生命周期。实例状态生存期由Android控制; Service
保持对象的持续时间由您控制。特别是如果CustomObject
可能很大,或者列表可能很长,我宁愿您更好地控制RAM消耗的时间。
答案 2 :(得分:1)
如果这主要是为了处理方向更改,可以Activity#onRetainNonConfigurationInstance()
做你想做的吗?
活动可以使用此API将广泛的状态从旧活动实例传播到新活动实例,从加载的位图传播到网络连接,以及均匀地主动运行的线程。请注意,您不应传播任何可能根据配置更改的数据,包括从字符串,布局或drawable等资源加载的任何数据。
如果您尝试的不仅仅是在配置更改中保留数据,那么此API将无法帮助您。
答案 3 :(得分:0)
据我所知,SavedInstanceState
旨在保存活动的UI配置(例如标准Android UI小部件,例如文本字段,会自动保存)。
如果要在不同的活动重新启动之间保存自定义对象(这不适用于用户通过单击后退按钮完成活动,但它确实适用于例如方向更改)。使用以下代码保留对象:
// maintain a reference to the EchoServer object when the activity is recreated
@Override
public Object onRetainNonConfigurationInstance() {
return <<your object of choice>>;
}
然后在onCreate(Bundle savedInstanceState)方法中,您可以检索该对象:
// if there is a saved instance state, restore the state
if (savedInstanceState != null) {
<<yourObject>> = (<<your object's class) getLastNonConfigurationInstance();
答案 4 :(得分:-2)
为什么不将对象保存到SD卡?您可以在我的博客&gt;&gt;上看到我如何使用此功能的示例http://androidworkz.com/2010/07/06/source-code-imageview-flipper-sd-card-scanner/
public void saveArray(String filename, String[] output_field) {
try {
FileOutputStream fos = new FileOutputStream(filename);
GZIPOutputStream gzos = new GZIPOutputStream(fos);
ObjectOutputStream out = new ObjectOutputStream(gzos);
out.writeObject(output_field);
out.flush();
out.close();
}
catch (IOException e) {
e.getStackTrace();
}
}
public String[] loadArray(String filename) {
try {
FileInputStream fis = new FileInputStream(filename);
GZIPInputStream gzis = new GZIPInputStream(fis);
ObjectInputStream in = new ObjectInputStream(gzis);
String[] read_field = (String[])in.readObject();
in.close();
return read_field;
}
catch (Exception e) {
e.getStackTrace();
}
return null;
}