2010-07-11 73 views
13

我有一个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 [email protected] 
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) 

有什么办法来存储自定义的情况下状态的对象?

回答

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 */}
+3

菲利克斯表示,他不能修改'CustomObject'。 – CommonsWare 2010-07-11 13:30:25

+0

@CommonsWare:谢谢。我没有注意到这一点。 – Macarse 2010-07-11 13:38:22

+0

我将此标记为已接受,因为这是我*将*去的路线。但是,我与图书馆的作者进行了交谈,他们将会实施它。 – Felix 2010-07-11 20:20:03

5

您的List<CustomObject>是否由服务持有,并通过本地绑定模式让您的活动可以访问您的服务。

您不仅不必担心在实例状态下保留它,而且您可以更好地控制内存中这些对象的生存期。实例状态生存期由Android控制; Service持有对象的时间由您控制。特别是如果CustomObject可能很大,或者列表可能很长,我宁愿你有更大的控制RAM的消耗时间。

+0

你的意思是有一个始终不停的“服务”? – Macarse 2010-07-11 13:44:58

+0

只有1+活动在运行时才会停止。一旦没有更多的连接,它将关闭。你也可以使用一个自定义的'Application'对象,如果你愿意的话 - 你失去了控制权,但没有一个“服务”说谎。 – CommonsWare 2010-07-11 13:48:57

+0

@CommonsWare:我猜他想'Serialize'来保存它以备后用。如果手机重新启动,使用该方法他将失去信息。 – Macarse 2010-07-11 14:04:46

1

如果这主要是为了处理方位变化,Activity#onRetainNonConfigurationInstance()可以做你想做的吗?

活动可以使用此API将广泛的状态从旧的活动实例传播到新的活动实例,从加载的位图到网络连接,以均匀主动运行线程。请注意,您不应该传播任何可能根据配置而改变的数据,包括从资源(例如字符串,布局或drawable)加载的任何数据。

如果您试图通过配置更改持久保存数据,此API无济于事。

+0

+1,我知道'onRetainNonConfigurationInstance()',但不,这不是主要用于方向更改 – Felix 2010-07-11 20:22:03

-2

为什么不把对象保存到SD卡?你可以看到如何使用这个在我的博客>>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; 
     } 
+1

因为...我的对象是自定义的,不能被序列化以便存储在SD卡上? – Felix 2010-07-13 11:43:53

+1

放松一下,一个downvote不是世界的尽头。 – jjnguy 2010-07-20 18:28:01

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(束savedInstanceState)方法,然后你可以检索对象:

// if there is a saved instance state, restore the state 
    if (savedInstanceState != null) { 
     <<yourObject>> = (<<your object's class) getLastNonConfigurationInstance(); 
+2

从API 13开始,onRetainNonConfigurationInstance方法已弃用。http://developer.android.com/reference/android/app/Activity.html#onRetainNonConfigurationInstance%28%29 – javahead76 2012-11-06 13:26:26

相关问题