如何正确地使用Parcelable和ArrayList<Parcelable>实现?

76

我在制作我的类Parcelable时遇到了困难。问题是,我尝试将该类的成员,它是一个ArrayList<Parcelable>对象写入包裹中。这个ArrayList是可序列化的,列表中的对象(ZigBeeDev)是Parcelable的。

以下是相关代码:

package com.gnychis.coexisyst;

import java.util.ArrayList;
import java.util.Iterator;

import android.os.Parcel;
import android.os.Parcelable;

public class ZigBeeNetwork implements Parcelable {

    public String _mac;             // the source address (of the coordinator?)
    public String _pan;             // the network address
    public int _band;               // the channel
    ArrayList<Integer> _lqis;       // link quality indicators (to all devices?)
    ArrayList<ZigBeeDev> _devices;  // the devices in the network

    public void writeToParcel(Parcel out, int flags) {
        out.writeString(_mac);
        out.writeString(_pan);
        out.writeInt(_band);
        out.writeSerializable(_lqis);
        out.writeParcelable(_devices, 0);  // help here
    }

    private ZigBeeNetwork(Parcel in) {
        _mac = in.readString();
        _pan = in.readString();
        _band = in.readInt();
        _lqis = (ArrayList<Integer>) in.readSerializable();
        _devices = in.readParcelable(ZigBeeDev.class.getClassLoader());  // help here
    }

    public int describeContents() {
        return this.hashCode();
    }

    public static final Parcelable.Creator<ZigBeeNetwork> CREATOR = 
            new Parcelable.Creator<ZigBeeNetwork>() {
        public ZigBeeNetwork createFromParcel(Parcel in) {
            return new ZigBeeNetwork(in);
        }

        public ZigBeeNetwork[] newArray(int size) {
            return new ZigBeeNetwork[size];
        }
    };

    //...
}

我已经标记了两个地方 "// help here",以便理解如何正确地写入包裹并如何重构它。如果ZigBeeDevParcelable(已经经过适当的测试),那么我该如何正确地执行这个操作?

5个回答

142

你已经快做对了!

你只需要做:

public void writeToParcel(Parcel out, int flags) {
    out.writeString(_mac);
    out.writeString(_pan);
    out.writeInt(_band);
    out.writeSerializable(_lqis);
    out.writeTypedList(_devices);
}

private ZigBeeNetwork(Parcel in) {
    _mac = in.readString();
    _pan = in.readString();
    _band = in.readInt();
    _lqis = (ArrayList<Integer>) in.readSerializable();
    in.readTypedList(_devices, ZigBeeDev.CREATOR);
}

就这些!

对于您的整数列表,您还可以进行以下操作:

out.writeList(_lqis);
_lqis = new ArrayList<>();
in.readList(_lqis Integer.class.getClassLoader());

它应该可以工作。


2
为了以后的参考,我不确定是否有人注意到了一个笔误,但是 ZigBeeDev.Creator 应该是 ZigBeeDev.CREATOR。在我自己的项目中,我每次都会遇到错误,直到我意识到有两个不同的创建者可以使用。 - gatlingxyz
@NitroG42 第二种读取列表的方法导致了“无法将void转换为ArrayList<Integer>”的错误。 - AlexAndro
8
我也需要在先实例化数组,例如我的代码看起来会像这样: _devices = new ArrayList<ZigBeeDev>(); in.readTypedList(_devices, ZigBeeDev.Creator); - sgarman
@sgarman 没有必要实例化数组,因为这样会出现警告:显式类型参数 ZigBeeDev 可以替换为 <>。此检查报告所有新的带类型参数的表达式,可以用钻石类型 <> 替换。 - crubio
确认使用通用的Hashtable,该Hashtable可以使用基本类型(包括String)作为键或值。 - Eido95

17

在我的情况下,in.readTypedList(_devices, ZigBeeDev.CREATOR);_devices 上给了我一个 NullPointerException。所以我使用了这个:

_devices = in.createTypedArrayList(ZigBeeDev.CREATOR);

3
如果你使用了in.readTypedList(),则需要先创建一个新的ArrayList(),然后将其传递给in.readTypedList(yourNewArrayList, YourObj.CREATOR)。否则它只会尝试把数据放到空列表中,这就是为什么你收到NPE的原因。FWIW表示"供参考"。 - rf43
是的,没错。在使用之前我已经阅读了源代码。谢谢你提供额外的信息,这是我忘记写的。 - osrl
@osrl 当我们按照您的解决方案操作时,出现了异常 java.lang.OutOfMemoryError: Failed to allocate a 13893804 byte allocation with 11004100 free bytes and 10MB until OOM。 - Ajit Kumar Dubey
你的数组存储文件或者其他大型数据吗?@Ajit - osrl

8

2
在构造函数中,您应该使用:
_lqis = in.createTypedArrayList(ZigBeeDev.CREATOR);

在“writeToParcel”中使用以下内容:
out.writeTypedList(_lqis);

0
有点晚了,但我也遇到了这个问题。 经过长时间的浪费,我偶然发现了parcelabler.com网站,它可以自动为您创建包裹。
它使我能够非常轻松地创建一个带有数组列表的嵌套包裹,并节省了很多时间。
基本上,该网站的工作方式是您输入带有ArrayList的对象,它会自动添加必要的方法以使其可包裹(自动生成读取包裹,写入包裹,描述内容和包裹创建者)。 在创建包含嵌套包裹,数组和列表等复杂包裹时,这特别有用。
编辑:此外,IntelliJ IDEA 和Android Studio都有适用于此的插件,与列出的网站类似。

这些插件基于类中的字段生成 Android Parcelable 样板代码。


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