Intent putExtra ArrayList<NameValuePair>

28

有人知道如何将 ArrayList<NameValuePair> 作为额外信息添加到 Intent 中吗?

ArrayList<NameValuePair> nameValuePairs = new ArrayList<NameValuePair>();
nameValuePairs.add(new BasicNameValuePair("first_name", first_name));
nameValuePairs.add(new BasicNameValuePair("last_name", last_name));
nameValuePairs.add(new BasicNameValuePair("email", email));
nameValuePairs.add(new BasicNameValuePair("password", password));

/* Move on to step 2 */
Intent intent = new Intent(RegisterActivity1.this, RegisterActivity2.class);
intent.putExtra("nvp", nameValuePairs);
startActivity(intent);

这是RegisterActivity2的类声明:

public class RegisterActivity2 extends Activity implements Serializable {
}

logcat 中的错误为:

> Parcel: unable to marshal value first_name="whatever"

在实施 Ted Hopp 提出的建议后,我仍然遇到错误。以下是错误堆栈跟踪:

08-04 22:10:16.095: E/AndroidRuntime(5065): FATAL EXCEPTION: main
> 08-04 22:10:16.095: E/AndroidRuntime(5065):
> java.lang.RuntimeException: Unable to start activity
> ComponentInfo{/.RegisterActivity2}: java.lang.RuntimeException:
> Parcelable encountered IOException reading a Serializable object (name
> = database.NVP) 08-04 22:10:16.095: E/AndroidRuntime(5065):   at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2070)
> 08-04 22:10:16.095: E/AndroidRuntime(5065):   at
> android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2095)
> 08-04 22:10:16.095: E/AndroidRuntime(5065):   at
> android.app.ActivityThread.access$600(ActivityThread.java:135) 08-04
> 22:10:16.095: E/AndroidRuntime(5065):     at
> android.app.ActivityThread$H.handleMessage(ActivityThread.java:1201)
> 08-04 22:10:16.095: E/AndroidRuntime(5065):   at
> android.os.Handler.dispatchMessage(Handler.java:99) 08-04
> 22:10:16.095: E/AndroidRuntime(5065):     at
> android.os.Looper.loop(Looper.java:137) 08-04 22:10:16.095:
> E/AndroidRuntime(5065):   at
> android.app.ActivityThread.main(ActivityThread.java:4849) 08-04
> 22:10:16.095: E/AndroidRuntime(5065):     at
> java.lang.reflect.Method.invokeNative(Native Method) 08-04
> 22:10:16.095: E/AndroidRuntime(5065):     at
> java.lang.reflect.Method.invoke(Method.java:511) 08-04 22:10:16.095:
> E/AndroidRuntime(5065):   at
> com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:795)
> 08-04 22:10:16.095: E/AndroidRuntime(5065):   at
> com.android.internal.os.ZygoteInit.main(ZygoteInit.java:562) 08-04
> 22:10:16.095: E/AndroidRuntime(5065):     at
> dalvik.system.NativeStart.main(Native Method) 08-04 22:10:16.095:
> E/AndroidRuntime(5065): Caused by: java.lang.RuntimeException:
> Parcelable encountered IOException reading a Serializable object (name
> = database.NVP) 08-04 22:10:16.095: E/AndroidRuntime(5065):   at android.os.Parcel.readSerializable(Parcel.java:2144) 08-04
> 22:10:16.095: E/AndroidRuntime(5065):     at
> android.os.Parcel.readValue(Parcel.java:2016) 08-04 22:10:16.095:
> E/AndroidRuntime(5065):   at
> android.os.Parcel.readListInternal(Parcel.java:2235) 08-04
> 22:10:16.095: E/AndroidRuntime(5065):     at
> android.os.Parcel.readArrayList(Parcel.java:1655) 08-04 22:10:16.095:
> E/AndroidRuntime(5065):   at
> android.os.Parcel.readValue(Parcel.java:1986) 08-04 22:10:16.095:
> E/AndroidRuntime(5065):   at
> android.os.Parcel.readMapInternal(Parcel.java:2226) 08-04
> 22:10:16.095: E/AndroidRuntime(5065):     at
> android.os.Bundle.unparcel(Bundle.java:223) 08-04 22:10:16.095:
> E/AndroidRuntime(5065):   at
> android.os.Bundle.getSerializable(Bundle.java:1254) 08-04
> 22:10:16.095: E/AndroidRuntime(5065):     at
> android.content.Intent.getSerializableExtra(Intent.java:4268) 08-04
> 22:10:16.095: E/AndroidRuntime(5065):     at
> .RegisterActivity2.onCreate(RegisterActivity2.java:24) 08-04
> 22:10:16.095: E/AndroidRuntime(5065):     at
> android.app.Activity.performCreate(Activity.java:5244) 08-04
> 22:10:16.095: E/AndroidRuntime(5065):     at
> android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1082)
> 08-04 22:10:16.095: E/AndroidRuntime(5065):   at
> android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2034)
> 08-04 22:10:16.095: E/AndroidRuntime(5065):   ... 11 more 08-04
> 22:10:16.095: E/AndroidRuntime(5065): Caused by:
> java.io.InvalidClassException:
> org.apache.http.message.BasicNameValuePair; IllegalAccessException
> 08-04 22:10:16.095: E/AndroidRuntime(5065):   at
> java.io.ObjectStreamClass.resolveConstructorClass(ObjectStreamClass.java:694)
> 08-04 22:10:16.095: E/AndroidRuntime(5065):   at
> java.io.ObjectStreamClass.newInstance(ObjectStreamClass.java:655)
> 08-04 22:10:16.095: E/AndroidRuntime(5065):   at
> java.io.ObjectInputStream.readNewObject(ObjectInputStream.java:1816)
> 08-04 22:10:16.095: E/AndroidRuntime(5065):   at
> java.io.ObjectInputStream.readNonPrimitiveContent(ObjectInputStream.java:787)
> 08-04 22:10:16.095: E/AndroidRuntime(5065):   at
> java.io.ObjectInputStream.readObject(ObjectInputStream.java:2003)
> 08-04 22:10:16.095: E/AndroidRuntime(5065):   at
> java.io.ObjectInputStream.readObject(ObjectInputStream.java:1960)
> 08-04 22:10:16.095: E/AndroidRuntime(5065):   at
> android.os.Parcel.readSerializable(Parcel.java:2142) 08-04
> 22:10:16.095: E/AndroidRuntime(5065):     ... 23 more
5个回答

24

正如其他人所指出的那样,当您想将数组作为意图附加项传递时,其元素需要实现 Serializable。在您的情况下,方法是 相当简单(不完全正确 — 请见下文):定义自己的 BasicNameValuePair 子类并声明它实现该接口。

public class NVP extends BasicNameValuePair implements Serializable {
    public NVP(String name, String value) {
        super(name, value);
    }
}

然后您可以像之前尝试的那样使用它:

ArrayList<NVP> nameValuePairs = new ArrayList<NVP>();
nameValuePairs.add(new NVP("first_name", first_name));
nameValuePairs.add(new NVP("last_name", last_name));
nameValuePairs.add(new NVP("email", email));
nameValuePairs.add(new NVP("password", password));

/* Move on to step 2 */
Intent intent = new Intent(RegisterActivity1.this, RegisterActivity2.class);
intent.putExtra("nvp", nameValuePairs);
startActivity(intent);

在接收端,您需要使用以下方法将其提取出来:

ArrayList<NVP> nameValuePairs = (ArrayList<NVP>) intent.getSerializable("nvp");

编辑 好了,上面的方法行不通,因为 BasicNameValuePair 没有默认构造函数,而序列化机制要求非可序列化的祖先类必须有一个默认构造函数。否则,在反序列化时会出现异常。

不幸的是,这意味着解决方案并不像我所描述的那么简单。一个解决方法是定义自己的类来实现 Parcelable,就像 @keerthana murugesan 在他的回答中建议的那样。如果你以前实现过 Parcelable,你就会知道它有点麻烦。另一个解决方法是定义一个包装 BasicNameValuePair 的类(而不是从它派生),并管理自己的序列化:

public class NVP implements NameValuePair, Serializable {
    private BasicNameValuePair nvp;

    public NVP(String name, String value) {
        nvp = new BasicNameValuePair(name, value);
    }

    @Override
    public String getName() {
        return nvp.getName();
    }

    @Override
    public String getValue() {
        return nvp.getValue();
    }

    // serialization support

    private static final long serialVersionUID = 1L;

    private void writeObject(ObjectOutputStream out) throws IOException {
        out.writeString(nvp.getName());
        out.writeString(nvp.getValue());
    }

    private void readObject(ObjectInputStream in) throws IOException, ClassNotFoundException {
        nvp = new BasicNameValuePair(in.readString(), in.readString());
    }

    private void readObjectNoData() throws ObjectStreamException {
        // nothing to do
    }
}

@usr55410 - 这并不完美。我建议子类化 NameValuePair,但这样做是行不通的。现在已经修复了。我想你自己也已经解决了这个问题,因为你已经接受了这个解决方案。 :) - Ted Hopp
@usr55410 - 哎呀,糟糕了。Serializable 机制要求每个祖先类都有一个默认构造函数,而 BasicNameValuePair 没有。让我想出一个应该可以工作的 NVP 类。 - Ted Hopp
@TedHopp - 又碰到你了!感谢你在这里帮助我解决了另一个问题:https://dev59.com/RYPba4cB1Zd3GeqPvaso。我想问如何使回调可序列化,因为我的回调出现了问题。我会尝试弄清楚如何提问而不被投票打击。 - Patricia
1
@Lucy - 在撰写你的问题之前,你可能会发现浏览这个关于LocalBroadcastReceiver这个关于序列化Bitmap的线程是值得的。使用本地广播是一种完全不同的方式来解耦你的POJO与应用程序的其他部分。 - Ted Hopp
@TedHopp - 哦,智者啊! 许多事情的大师! 你又救了我一次。 谢谢你的技巧......有效了。 万岁你!! :-) - Patricia
显示剩余3条评论

23

我遇到了同样的问题,尝试在对象类中不实现Serializable接口的情况下发送ArrayList中的对象。

一个简单的解决方法是将ArrayList封装在Bundle中。

因此,在第一个活动中 ->

ArrayList<Object> objects = new ArrayList<Object>(); 
// Object class does not implement Serializable interface

Bundle extra = new Bundle();
extra.putSerializable("objects", objects);

Intent intent = new Intent(getBaseContext(), ShowSpread.class);
intent.putExtra("extra", extra);

第二个活动 ->

Bundle extra = getIntent().getBundleExtra("extra");
ArrayList<Object> objects = (ArrayList<Object>) extra.getSerializable("objects")

运作良好。


1
为什么这不是正确答案?它很简单易懂。 - Zen
我正在使用这个。但是你遇到过什么缺点吗?或者实现Serializable有什么优势吗? - Zen
@Zen 意思是只需要在你的对象类中加入 implements Serializable 即可。 - Taufik Nur Rahmanda

1
你需要将一个对象数组作为可序列化对象传递到意图中。该对象数组必须由内部也是可序列化的对象组成。
ArrayList<NameValuePair> nameValuePairs = new ArrayList<NameValuePair>();
nameValuePairs.add(new BasicNameValuePair("first_name", first_name));
nameValuePairs.add(new BasicNameValuePair("last_name", last_name));
nameValuePairs.add(new BasicNameValuePair("email", email));
nameValuePairs.add(new BasicNameValuePair("password", password));

在这里,BaseNameValuePair需要是一个实现了Serializable的类。使用ArrayListtoArray方法将其放入put中,并进行Serializable的强制类型转换。在您的getter Activity中,使用意图的getSerializable方法。

0

可以使用 Parcelable 接口来实现。

  1. 创建你的 BasicNameValuePair 类并实现 Parcelable 接口。可以参考this

    public class BasicNameValuePair implements Parcelable{
    
    }
    
  2. 像之前一样,将这些对象添加到一个 ArrayList 中:

    ArrayList<NameValuePair> nameValuePairs = new ArrayList<NameValuePair>();
    nameValuePairs.add(new BasicNameValuePair("first_name", first_name));
    nameValuePairs.add(new BasicNameValuePair("last_name", last_name));
    nameValuePairs.add(new BasicNameValuePair("email", email));
    nameValuePairs.add(new BasicNameValuePair("password", password));
    
  3. 使用 putParcelableArrayListExtra() 将它们传递给 intent

    intent.putParcelableArrayListExtra ("nvp", nameValuePairs);
    startActivity(intent);
    

0

为了使其正常工作,您的列表所持有的泛型类型必须实现Serializable接口。


谢谢。我已经在上面添加了下一个活动的类声明。这是你所指的吗?但我仍然得到相同的错误。 - AndroidDev
不,你的List所持有的类是BasicNameValuePair,它没有实现Serializable接口。这就是你的问题所在。 - Karakuri

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接