Problem unmarshalling parcelables
Asked Answered
A

10

61

I've got a few classes that implement Parcelable and some of these classes contain each other as properties. I'm marshalling the classes into a Parcel to pass them between activities. Marshalling them TO the Parcel works fine, but when I try to unmarshall them I get the following error:

...
AndroidRuntime  E  Caused by: android.os.BadParcelableException: ClassNotFoundException when unmarshalling: schemas.Arrivals.LocationType
AndroidRuntime  E   at android.os.Parcel.readParcelable(Parcel.java:1822)
AndroidRuntime  E   at schemas.Arrivals.LayoverType.<init>(LayoverType.java:121)
AndroidRuntime  E   at schemas.Arrivals.LayoverType.<init>(LayoverType.java:120)
AndroidRuntime  E   at schemas.Arrivals.LayoverType$1.createFromParcel(LayoverType.java:112)
AndroidRuntime  E   at schemas.Arrivals.LayoverType$1.createFromParcel(LayoverType.java:1)
AndroidRuntime  E   at android.os.Parcel.readTypedList(Parcel.java:1509)
AndroidRuntime  E   at schemas.Arrivals.BlockPositionType.<init>(BlockPositionType.java:244)
AndroidRuntime  E   at schemas.Arrivals.BlockPositionType.<init>(BlockPositionType.java:242)
AndroidRuntime  E   at schemas.Arrivals.BlockPositionType$1.createFromParcel(BlockPositionType.java:234)
AndroidRuntime  E   at schemas.Arrivals.BlockPositionType$1.createFromParcel(BlockPositionType.java:1)
...

The LayoverType class (where it's failing):

public class LayoverType implements Parcelable {    
    protected LocationType location;
    protected long start;
    protected long end;

    public LayoverType() {}

    public LocationType getLocation() {
        return location;
    }

    public void setLocation(LocationType value) {
        this.location = value;
    }

    public long getStart() {
        return start;
    }

    public void setStart(long value) {
        this.start = value;
    }

    public long getEnd() {
        return end;
    }

    public void setEnd(long value) {
        this.end = value;
    }


    // **********************************************
    //  for implementing Parcelable
    // **********************************************

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

    @Override
    public void writeToParcel(Parcel dest, int flags) {
        dest.writeParcelable(location, flags);
        dest.writeLong(start);
        dest.writeLong(end  );
    }

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

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

    private LayoverType(Parcel dest) {
        location = (LocationType) dest.readParcelable(null);    // it's failing here
        start = dest.readLong();
        end   = dest.readLong();
    }
}

Here's the LocationType class:

public class LocationType implements Parcelable {
    protected int locid;
    protected String desc;
    protected String dir;
    protected double lat;
    protected double lng;

    public LocationType() {}

    public int getLocid() {
        return locid;
    }

    public void setLocid(int value) {
        this.locid = value;
    }

    public String getDesc() {
        return desc;
    }

    public void setDesc(String value) {
        this.desc = value;
    }

    public String getDir() {
        return dir;
    }

    public void setDir(String value) {
        this.dir = value;
    }

    public double getLat() {
        return lat;
    }

    public void setLat(double value) {
        this.lat = value;
    }

    public double getLng() {
        return lng;
    }

    public void setLng(double value) {
        this.lng = value;
    }

    // **********************************************
    //  for implementing Parcelable
    // **********************************************


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

    @Override
    public void writeToParcel(Parcel dest, int flags) {
        dest.writeInt   (locid);
        dest.writeString(desc );
        dest.writeString(dir  );
        dest.writeDouble(lat  );
        dest.writeDouble(lng  );
    }

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

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

    private LocationType(Parcel dest) {
        locid = dest.readInt   ();
        desc  = dest.readString();
        dir   = dest.readString();
        lat   = dest.readDouble();
        lng   = dest.readDouble();
    }
}

Update 2: As far as I can tell it's failing at the following bit of code (from Parcel's source):

Class c = loader == null ? Class.forName(name) : Class.forName(name, true, loader);

Why is it not able to find the class? It both exists and implements Parcelable.

Angadresma answered 3/1, 2010 at 21:2 Comment(7)
Could you post the code for the LocationType class? It could be the write method in that class is not correct in some way.Kootenay
@MpKamowski: I added the Parcelable bits from LocationType. The only other stuff in either class is just getters/setters.Angadresma
Is it possible that the package for which LocationType is a apart of is not visible in the code where it is being unmarshalled?Kootenay
I can't see how. LocationType and LayoverType are in the same package. I went ahead and tried implicitly importing LocationType into LayoverType and it made no difference.Angadresma
Could you post the full code for both of your classes? I'm especially interested in LocationType to see whether it's a public static inner class or not.Banded
I went ahead and included everything but the package line and the imports. Hope that helps and thanks for taking a look at this.Angadresma
possible duplicate of "BadParcelableException: ClassNotFoundException when unmarshalling <myclass>" while using Parcel.read method that has a ClassLoader as argumentCamber
L
82

Because this was not answered in "answer" but in comment I will post an answer: As @Max-Gontar pointed you should use LocationType.class.getClassLoader() to get the correct ClassLoader and get rid of ClassNotFound exception, i.e.:

in.readParceleable(LocationType.class.getClassLoader());

Lethe answered 6/2, 2011 at 18:25 Comment(1)
This was so helpful! In my case, I had to pass in the class loader when unmarshalling a Parcelable object that contained a bundle that also contained instances of the same type of object. Ex. public class MyObject implements Parcelable, Serializable { protected Bundle mBundle; MyObject(Parcel parcel) { mBundle = parcel.readBundle(MyObject.class.getClassLoader()); } }Brigitta
C
36

I had the same problem with the following setup: some handler creates a Message and sends its over a Messenger to a remote service.

the Message contains a Bundle where I put my Parcelable descendant:

final Message msg = Message.obtain(null, 0);
msg.getData().putParcelable("DOWNLOADFILEURLITEM", downloadFileURLItem);

messenger.send(msg);

I had the same exception when the remote service tried to unparcel. In my case, I had overseen that the remote service is indeed a separate os process. Therefore, I had to set the current classloader to be used by the unparcelling process on the service side:

final Bundle bundle = msg.getData();
bundle.setClassLoader(getClassLoader());

DownloadFileURLItem urlItem = (DownloadFileURLItem)
bundle.getParcelable("DOWNLOADFILEURLITEM");

Bundle.setClassLoader sets the classloader which is used to load the appropriate Parcelable classes. In a remote service, you need to reset it to the current class loader.

Chromo answered 27/9, 2010 at 18:41 Comment(0)
P
9

I found the problem was I was not passing my applications ClassLoader to the unmarshalling function:

in.readParceleable(getContext().getClassLoader());

Rather than:

in.readParceleable(null); 

OR

in.readParceleable(MyClass.class.getClassLoader()); 
Pirali answered 3/3, 2010 at 12:22 Comment(5)
I suppose you mean LocationType.class.getClassLoader()Lewis
That looked hopeful, but does not solve my problem. Can you please post the minimal project that illustrates the problem and its solution? I am struggling with the sample code from Google and it crashes in the receiver-activity.Gent
@MaxGontar yes probably. But it's so long since I wrote this I don't want to edit it in case it is wrong.Pirali
I used MyApplication.class.getClassLoader() as we were using generics for a parcelable object, worked fine.Swamper
@MaxHowell i got the same issue, I made my own Parcelable object which has a HashMap<String, String> to save the columnsNames and its values. My Parcelable Object save all the data in the Parce object in the writeToParcelMethod(Parcel dest, int flags) and retrieve them in the readFromParcel(Parcel parcel)` method. after i put all Parcelable objects inside of ArrayList<ParcelableRow> and i send it(from a remote process) to an activity by Messenger object. even if i do data.setClassLoader(getClassLoader()) but when i try to populate the parcelable objects from the list they are null. thkUttasta
G
8

Just adding my 2 cents here, because I lost more than half a day scratching my head on this. You might get this error if you don't put the writes methods and reads methods in the exact same order. For instance the following would be wrong:

 @Override
    // Order: locid -> desc -> lat -> dir -> lng
    public void writeToParcel(Parcel dest, int flags) {
        dest.writeInt   (locid);
        dest.writeString(desc);
        dest.writeDouble(lat);
        dest.writeString(dir);
        dest.writeDouble(lng);
    }
    // Order: locid -> desc -> dir -> lat -> lng
    private LocationType(Parcel dest) {
        locid = dest.readInt();
        desc  = dest.readString();
        dir   = dest.readString();
        lat   = dest.readDouble();
        lng   = dest.readDouble();
    }

By the way the author did this correctly but it might help someone one day.

Gatepost answered 9/11, 2011 at 9:16 Comment(3)
And also, don't do something like writeInt writeLong and then readInt readInt on the other side.Np
Helped me, thanks! Made a stupid mistake and wasted a few hours because of this...error messages didn't really help much.Makalu
Right Answer. It's simple one. But half day gone.Copulation
K
6

I am not very familiar with Parcelable but if it's anything like Serialization each call to write an object that implements the interface will cause a recursive call to writeToParcel(). Therefore, if something along the call stack fails or writes a null value the class that initiated the call may not be constructed correctly.

Try: Trace the writeToParcel() call stack through all the classes starting at the first call to writeToParcel() and verify that all the values are getting sent correctly.

Kootenay answered 3/1, 2010 at 21:35 Comment(6)
I was definately having problems with NullPointers before, but I initialized all the values and that went away. Now it seems like it just can't find the class.Angadresma
Oh, it appears you were right all along. As best I can tell it was crapping out because something wasn't getting initialized and when it tried to read the next Parcellable out it was getting a null (or something).Angadresma
can you tell us what exactly you fixed? I'm having the same problem and don't see what's wrong with my code.Sokil
I also fixed it, here's what I learned from my own mistakes (and the atrocious documentation from Binder/Parcel/Parcelable): 1) you have to pair calls to write/read something exactly (i.e. you cannot use writeValue() to write a Parcelable and readParcelable() to read it). 2) Don't use readTypedList(), it always broke on me. Instead, createTypedArrayList() is much eaiser to use and also worked as expected. 3) on every call to readXYZ where it expects a ClassLoader, pass that class' loader, not null. That's also for readBundle() when it must unparcel values in its internal map. HTH.Sokil
Is there any chance that someone knows of a working implementation example of this?Carob
using ClassLoader instead of null inside readList() did the trick! thanks!Toggery
A
6

I got ClassNotFoundException too and posting my solution bc the answers here led me to the right direction. My scenario is that I have nested parcelable objects. Object A contains an ArrayList of Object's B. Both implement Parcelable. Writing the list of B Object's in class A:

@Override
public void writeToParcel(Parcel dest, int flags) {
    ...
    dest.writeList(getMyArrayList());

}

Reading the list in class A:

public ObjectA(Parcel source) {
    ...
    myArrayList= new ArrayList<B>();
    source.readList(myArrayList, B.class.getClassLoader());
}

Thank you!

Auxin answered 12/2, 2013 at 11:15 Comment(0)
D
5

Instead of using writeParcelable and readParcelable use writeToParcel and createFromParcel directly. So the better code is:

@Override
public void writeToParcel(Parcel dest, int flags) {
    location.writeToParcel(dest, flags);
    dest.writeLong(start);
    dest.writeLong(end  );
}

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

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

private LayoverType(Parcel dest) {
    location = LocationType.CREATOR.createFromParcel(dest);
    start = dest.readLong();
    end   = dest.readLong();
}
Dioxide answered 26/12, 2010 at 23:23 Comment(1)
How would you handle the case that location is null?Caspar
C
2

Well I had the same problem and solve it in a very silly way that I dont know if its called a solution at all.

lets say you have this class you want to pass to another activity

public class Person implements  Parcelable,Serializable{
public  String Name;
public  int Age;

@Override
public void writeToParcel(Parcel dest, int flags) {

 dest.writeString(name);
dest.writeInt(age);

}

public SeriesInfo(Parcel in) {

age= in.readInt(); //her was my problem as I have put age befor name
//while in the writeToParcel function I have defined
//dest.writeInt(age) after  in.readString();???!!!!
name= in.readString();

}
}

Thats it When I changed the:

dest.writeString(name);
dest.writeInt(age);

to

dest.writeInt(age);
dest.writeString(name);

The problem was solved???!!!!

Coeternal answered 5/1, 2016 at 2:59 Comment(1)
You have to read the values out in the same order you wrote them in.Angadresma
B
0

If you have an Object with a property of type of List objects you should pass the class loader when you read the property for example:

public class Mall implements Parcelable {

    public List<Outstanding> getOutstanding() {
        return outstanding;
    }

    public void setOutstanding(List<Outstanding> outstanding) {
        this.outstanding = outstanding;
    }        

    protected Mall(Parcel in) {
        outstanding = new ArrayList<Outstanding>();
        //this is the key, pass the class loader
        in.readList(outstanding, Outstanding.class.getClassLoader());
    }

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

    @Override
    public void writeToParcel(Parcel dest, int flags) {
        dest.writeList(outstanding);
    }

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

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

Note: Is important that the class Outstanding implements the Parceable interface.

Brandenburg answered 21/11, 2013 at 21:59 Comment(1)
ten years alter you could use parcel.createTypedArrayList for this.Ishmaelite
S
0

I got this exception because I was missing a constructor. The same must be done for all classes that implement Parcelable:

// add new constructor
@RequiresApi(Build.VERSION_CODES.N)
private LocationType(Parcel dest, ClassLoader loader) {
    super(dest, loader);
    locid = dest.readInt();
    desc  = dest.readString();
    dir   = dest.readString();
    lat   = dest.readDouble();
    lng   = dest.readDouble();
}

public static final Creator<LayoverType> CREATOR = new ClassLoaderCreator<LayoverType>() {

    // add createFromParcel method with ClassLoader
    @Override
    public LayoverType createFromParcel(Parcel in, ClassLoader loader)
    {
        return Build.VERSION.SDK_INT >= Build.VERSION_CODES.N ? new LayoverType(in, loader) : new LayoverType(in);
    }

    public LayoverType createFromParcel(Parcel in) {
        // call other createFromParcel method.    
        return createFromParcel(in, null);
    }

    public LayoverType[] newArray(int size) {
        return new LayoverType[size];
    }
};
Soapbox answered 12/12, 2019 at 7:47 Comment(0)

© 2022 - 2024 — McMap. All rights reserved.