在实例状态中保存自定义对象数组

时间:2010-07-11 12:25:37

标签: java android serialization

我有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)

有没有办法在实例状态中存储自定义对象?

5 个答案:

答案 0 :(得分:6)

制作CustomObject工具Parcelable并使用:

outState.putParcelable(KEY, myList);
onSaveInstanceState(outState);

同时检查this教程。

CommonsWare评论后

编辑

如果您的CustomObject未实现SerializableParcelable,我会尝试将其包装在您自己的对象中并添加:

  • 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;
      }