як правильно реалізувати Parcelable за допомогою ArrayList <Parcelable>?


76

У мене проблеми із складанням мого класу Parcelable. Проблема в тому, що я намагаюся написати на посилку учасника класу, який є ArrayList<Parcelable>об’єктом. ArrayListЦе Serializable, і об'єкти ( 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];
        }
    };

    //...
}

Я позначив два місця "// допоможи тут", щоб зрозуміти, як правильно писати на посилку та як її реконструювати. Якщо ZigBeeDevце Parcelable(належним чином перевірено), як мені це зробити належним чином?

Відповіді:


143

Ви майже зрозуміли!

Вам просто потрібно зробити:

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 Другий підхід до результатів списку прочитанихcannot convert from void to ArrayList<Integer>
AlexAndro

8
Мені також також потрібно було створити екземпляр масиву, наприклад, мій код мав би виглядати так: _devices = new ArrayList <ZigBeeDev> (); in.readTypedList (_devices, ZigBeeDev.Creator);
sgarman

@sgarman Немає потреби створювати екземпляр причини масиву, тоді ви отримуєте попередження: Явний аргумент типу ZigBeeDev можна замінити на <> Ця інспекція повідомляє про всі нові вирази з аргументами типу, які можна замінити на тип алмазу <>
crubio

Підтвердьте, що ви працюєте із загальним, Hashtableякий використовує примітивні типи (включаючи String) для будь-якого з його ключів або значень.
Eido95,

17

У моєму випадку in.readTypedList(_devices, ZigBeeDev.CREATOR);дав мені NullPointerExceptionON _devices. Тому я використав це:

_devices = in.createTypedArrayList(ZigBeeDev.CREATOR);

3
fwiw - причина, через яку ви отримували NPE, полягає в тому, що якщо ви використовуєте, in.readTypedList()вам потрібно спочатку створити новий ArrayList (), а потім передати його in.readTypedList(yourNewArrayList, YourObj.CREATOR)іншому, інакше він буде намагатися лише помістити дані в нульовий список.
rf43

Так, це правильно. Я прочитав вихідний код, перш ніж використовувати його. Дякую за додаткову інформацію, яку я забув написати.
osrl

@osrl, коли ми слідуємо за вашим рішенням, отримав виняток java.lang.OutOfMemoryError: Не вдалося розподілити розподіл байтів 13893804 з 11004100 безкоштовними байтами та 10 МБ до OOM
Ajit Kumar Dubey

Ваш масив містить файли чи щось велике? @Ajit
osrl


2

У конструкторі ви повинні використовувати

_lqis = in.createTypedArrayList(ZigBeeDev.CREATOR);

А в "writeToParcel" використовуйте

out.writeTypedList(_lqis);

0

Трохи пізно, але у мене теж була ця проблема. Після довгої втрати часу я натрапив на веб-сайт parcelabler.com, який автоматично створює посилки для вас.

Це дозволило мені дуже легко створити вкладену посилку зі списком масивів і заощадило мені багато часу.

В основному, як працює веб-сайт, ви вводите свій об’єкт із ArrayList, і він автоматично додає необхідні методи, щоб зробити його парцеляційним (читати з посилки, писати на посилку, описувати вміст та Creator parcel генерується автоматично). Це особливо корисно при створенні складних посилок, таких як запитання тут, які містять вкладені посилки, масиви та списки.

РЕДАГУВАТИ: Також IntelliJ IDEA та Android Studio мають плагіни для цього, які роблять подібне до перерахованого веб-сайту:

Ці плагіни генерують шаблонний код Android Parcelable на основі полів класу.

Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.