Android Parcelable 排序

4
我遇到了一些与 Parcelable 自定义类有关的问题,这些问题是在我为 Android 应用程序创建时出现的,但我已经以一种非常奇怪的方式解决了这些问题。
只在几种情况下读取 Parcelable 时会发生崩溃(这使我认为我的实现并不完全错误)。
java.lang.RuntimeException: Unable to start activity ComponentInfo{com.worldcraze.worldcraze/com.worldcraze.worldcraze.AdActivity}: android.os.BadParcelableException: ClassNotFoundException when unmarshalling: Surface Book
at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2416)
at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2476)
at android.app.ActivityThread.-wrap11(ActivityThread.java)
at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1344)
at android.os.Handler.dispatchMessage(Handler.java:102)
at android.os.Looper.loop(Looper.java:148)
at android.app.ActivityThread.main(ActivityThread.java:5417)
at java.lang.reflect.Method.invoke(Native Method)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:726)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:616)

Caused by: android.os.BadParcelableException: ClassNotFoundException when     unmarshalling: Surface Book
at android.os.Parcel.readParcelableCreator(Parcel.java:2432)
at android.os.Parcel.readParcelable(Parcel.java:2358)
at com.worldcraze.worldcraze.API.Model.TransportOffer.<init>     (TransportOffer.java:33)
at     com.worldcraze.worldcraze.API.Model.TransportOffer$1.createFromParcel(TransportO    ffer.java:43)
    at     com.worldcraze.worldcraze.API.Model.TransportOffer$1.createFromParcel(TransportO    ffer.java:40)
at android.os.Parcel.createTypedArray(Parcel.java:2167)
at com.worldcraze.worldcraze.API.Model.Ad.<init>(Ad.java:42)
at com.worldcraze.worldcraze.API.Model.Ad$1.createFromParcel(Ad.java:52)
at com.worldcraze.worldcraze.API.Model.Ad$1.createFromParcel(Ad.java:49)
at android.os.Parcel.readParcelable(Parcel.java:2367)
at android.os.Parcel.readValue(Parcel.java:2264)
at android.os.Parcel.readArrayMapInternal(Parcel.java:2614)
at android.os.BaseBundle.unparcel(BaseBundle.java:221)
at android.os.Bundle.getParcelable(Bundle.java:786)
at android.content.Intent.getParcelableExtra(Intent.java:5377)
at com.worldcraze.worldcraze.AdActivity.onCreate(AdActivity.java:57)
at android.app.Activity.performCreate(Activity.java:6251)
at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1107)
at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2369)
... 9 more

这是我原始的模型实现(Ad.java),崩溃发生在其中。
public class Ad implements Parcelable {

public String              author;
public String              transporter;
public Image               cover_urls;
public String              details;
public String              id;
public int                 item_size;
public String              object_name;
public Money               tips;
public Money               price;
public Location            where_to_buy;
public String              resource_uri;
public Location            where_to_deliver;
public Permissions         permissions;
public TransportOffer[]    transport_offers;
public float               fees_lemonway_CB;
public String              validation_code;
public Date                creation_dtime;
public String              accepted_transport_offer_id;

protected   Ad(Parcel in) {
    author           = in.readString();
    transporter      = in.readString();
    details          = in.readString();
    id               = in.readString();
    item_size        = in.readInt();
    object_name      = in.readString();
    resource_uri     = in.readString();
    cover_urls       = in.readParcelable(Image.class.getClassLoader());
    tips             = in.readParcelable(Money.class.getClassLoader());
    price            = in.readParcelable(Money.class.getClassLoader());
    permissions      = in.readParcelable(Permissions.class.getClassLoader());
    transport_offers = in.createTypedArray(TransportOffer.CREATOR);
    where_to_buy     = in.readParcelable(Location.class.getClassLoader());
    where_to_deliver = in.readParcelable(Location.class.getClassLoader());
    fees_lemonway_CB = in.readFloat();
    validation_code  = in.readString();
    creation_dtime   = (Date) in.readSerializable();
    accepted_transport_offer_id = in.readString();
}

public static final Creator<Ad> CREATOR = new Creator<Ad>() {
    @Override
    public Ad createFromParcel(Parcel in) {
        return new Ad(in);
    }

    @Override
    public Ad[] newArray(int size) {
        return new Ad[size];
    }
};

@Override
public int describeContents() {
    return 0;
}

@Override
public void writeToParcel(Parcel dest, int flags) {
    dest.writeString(author);
    dest.writeString(transporter);
    dest.writeString(details);
    dest.writeString(id);
    dest.writeInt(item_size);
    dest.writeString(object_name);
    dest.writeString(resource_uri);
    dest.writeParcelable(cover_urls, flags);
    dest.writeParcelable(tips, flags);
    dest.writeParcelable(price, flags);
    dest.writeParcelable(permissions, flags);
    dest.writeTypedArray(transport_offers, flags);
    dest.writeParcelable(where_to_buy, flags);
    dest.writeParcelable(where_to_deliver, flags);
    dest.writeFloat(fees_lemonway_CB);
    dest.writeString(validation_code);
    dest.writeSerializable(creation_dtime);
    dest.writeString(accepted_transport_offer_id);
}

我通过改变某些Parcelable属性的读写顺序来解决了问题。现在我在cover_urls之前读/写where_to_buy和where_to_deliver,这似乎已经解决了问题。

    where_to_buy     = in.readParcelable(Location.class.getClassLoader());
    where_to_deliver = in.readParcelable(Location.class.getClassLoader());
    cover_urls       = in.readParcelable(Image.class.getClassLoader());
    tips             = in.readParcelable(Money.class.getClassLoader());
    price            = in.readParcelable(Money.class.getClassLoader());
    permissions      = in.readParcelable(Permissions.class.getClassLoader());
    transport_offers = in.createTypedArray(TransportOffer.CREATOR);

在writeToParcel中的顺序是相同的,我只是为了不复制粘贴而省略了几行代码。

这个奇怪的修复方法非常有效,但我不知道原因。

有人遇到过类似的问题或知道打包/解包的顺序如何影响结果吗?

干杯!

1个回答

8

这不是异常行为。如果你了解其内部工作原理,你就会明白为什么会出现异常。

序列化基本上是按顺序写入字节。所以你要按某种顺序(例如4-16-1-2)写入一些可变长度的字节,然后从Parcel读取字节。

在从Parcel读取时,你尝试读取例如16个字节,但实际上你只写了4个字节,这就是崩溃发生的原因。

读取的顺序应该与写入的顺序相匹配,否则你将面临这样的情况:尝试读取例如16个字节,但实际上你的变量只有4个字节。


我有一个问题,假设我想通过Intent将一个listener对象传递到另一个Activity。现在我希望在Activity旋转时持久化该对象。我从onRestoreInstanceState()中的bundle的Parcel中获取的对象与之前的对象是否相同?使用Parcelable接口获取的对象是相同的吗?还是只是使用先前相同的值(字段变量)获得新对象? - Sayan Mukherjee
@SayanMukherjee,你不应该将监听器对象从一个活动传递到另一个活动。Bundle 不打算以这种方式使用。 - azizbekian
那不是我问的!我问的是,“我们是否通过使用Parcelable接口获得相同的对象?还是只是使用先前的相同值(字段变量)获得一个新对象?” - Sayan Mukherjee

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