Unmarshalling errors in Android app with custom parcelable classes
Solution 1
Android has two different classloaders: the framework classloader (which knows how to load Android classes) and the APK classloader (which knows how to load your code). The APK classloader has the framework classloader set as its parent, meaning it can also load Android classes.
Error #2 is likely caused by the Bundle using the framework classloader so it doesn't know of your classes. I think this can happen when Android needs to persist your Bundle and later restore it (for example when running out of memory in the background). You can fix this by setting the APK classloader on the bundle:
savedInstanceState.setClassLoader(getClass().getClassLoader());
Error #1 and #3 are more mysterious, are you perhaps writing null values in writeToParcel()
? Android doesn't like that very much I'm afraid.
Solution 2
Before reading from bundle, set ClassLoader:
bundle.setClassLoader(getClass().getClassLoader());
This saved my time!
Solution 3
By the look of it, the createFromParcel
and newArray
should be overridden like this:
public static final Parcelable.Creator<MyObject> CREATOR = new Parcelable.Creator<MyObject>() {
@Override
public MyObject createFromParcel(Parcel in) {
MyObject myObj = new MyObject();
myObj.intArray = in.readIntArray(...);
myObj.intValue = in.readInt(...);
// ....
// IN THE SAME ORDER THAT IS WRITTEN OUT AS PER writeToParcel!
//
return myObj;
}
@Override
public MyObject[] newArray(int size) {
return new MyObject[size];
}
};
Edit:
I forgot to mention that for the above to work, there should have been an empty constructor!
public MyObject(){}
caw
Updated on June 15, 2022Comments
-
caw about 2 years
For my Android application, I get several unmarshalling errors although I think I've done everything that is needed to properly save and load objects via
Parcelable
s. Can you tell me what's wrong with my code?Error 1:
java.lang.RuntimeException: Unable to start activity ComponentInfo Caused by: java.lang.RuntimeException: Parcel android.os.Parcel@41279860: Unmarshalling unknown type code 6619241 at offset 1372 at android.os.Parcel.readValue(Parcel.java:1922) at android.os.Parcel.readMapInternal(Parcel.java:2094) at android.os.Bundle.unparcel(Bundle.java:223) at android.os.Bundle.getParcelable(Bundle.java:1158) at android.app.Activity.onCreate(Activity.java:860) at my.app.package.PlayComputer.onCreate(PlayComputer.java:1012) at android.app.Activity.performCreate(Activity.java:4465)
Line 1012 in
MyActivity
is the call tosuper.onCreate(savedInstanceState);
in theActivity
'sonCreate()
.protected void onSaveInstanceState(Bundle savedInstanceState) { if (myObject == null) { savedInstanceState.putParcelable("myObject", null); } else { savedInstanceState.putParcelable("myObject", myObject); } savedInstanceState.putInt(...); savedInstanceState.putString(...); savedInstanceState.putBoolean(...); super.onSaveInstanceState(savedInstanceState); }
myObject
is of classMyObject
which has the following methods:public void writeToParcel(Parcel out, int flags) { out.writeIntArray(...); out.writeInt(...); out.writeStringArray(...); out.writeString(...); out.writeParcelableArray(..., flags); } public static final Parcelable.Creator<MyObject> CREATOR = new Parcelable.Creator<MyObject>() { public MyObject createFromParcel(Parcel in) { try { if (in == null) { return null; } else { return new MyObject(in); } } catch (Exception e) { return null; } } public MyObject[] newArray(int size) { return new MyObject[size]; } }; private MyObject(Parcel in) { in.readIntArray(...); ... = in.readInt(); in.readStringArray(...); ... = in.readString(); ... = (OtherObject[]) in.readParcelableArray(OtherObject.class.getClassLoader()); }
Error 2:
java.lang.RuntimeException: Unable to start activity ComponentInfo Caused by: android.os.BadParcelableException: ClassNotFoundException when unmarshalling: at android.os.Parcel.readParcelable(Parcel.java:1971) at android.os.Parcel.readValue(Parcel.java:1859) at android.os.Parcel.readMapInternal(Parcel.java:2099) at android.os.Bundle.unparcel(Bundle.java:223) at android.os.Bundle.getParcelable(Bundle.java:1158) at android.app.Activity.onCreate(Activity.java:905) at my.app.package.PlayComputer.onCreate(SourceFile:1012)
Same files and classes.
Error 3:
java.lang.RuntimeException: Unable to start activity ComponentInfo Caused by: java.lang.RuntimeException: Parcel android.os.Parcel@4051aff8: Unmarshalling unknown type code 7340149 at offset 1276 at android.os.Parcel.readValue(Parcel.java:1913) at android.os.Parcel.readMapInternal(Parcel.java:2083) at android.os.Bundle.unparcel(Bundle.java:208) at android.os.Bundle.getParcelable(Bundle.java:1100) at my.app.package.PlayComputer.onCreate(SourceFile:1111)
This time, the causing line (1111) is the following one:
myObject = (MyObject) savedInstanceState.getParcelable("myObject");
-
caw over 11 yearsThank you! Isn't this exactly what I am doing? I'm just using the constructor
MyObject(in)
to make things more clear. Ah okay, maybe you didn't see this because of my typo in the constructor's name, which has been corrected now. -
t0mm13b over 11 yearsThe only thing I can think of, is perhaps there should be an empty constructor, have amended the answer...
-
t0mm13b over 11 yearsplease verify that you are
@Override
the two methodsdescribeContents()
andwriteToParcel()
:) -
caw over 11 yearsI've checked that I'm overriding these methods - and yes, I do. Don't you see that your solution and my not-really-working solution are exactly the same? I've just put some code into a helper method called
GameState(Parcel in)
- just like in the documentation. -
t0mm13b over 11 yearsThat is how I use it in my projects and have no issues with
Parcelable
- sorry if my answer is not of much help to you. -
caw over 11 yearsNo problem ;) It's great that it works for your, but as there is no difference to my solution, it doesn't really help. Probably the cause for the problems is reading and writing arrays (of parcelable objects) and you don't have that, so your code works for you.
-
t0mm13b over 11 yearsHave used reading/writing arrays btw, its obviously something silly you're doing wrong somewhere for that to blow up in your face with the exception... try printing it/logcat it to see where is it going..from initialization, in every method to see where is it actually going wrong - that will help you nail it :) protip I use logcat.d(TAG, "if (....){ *** HERE ***}else{}"); in every statement flow to verify! Just use your imagination that suits yourself ;)
-
Stefan de Bruijn over 11 yearsMarco W. I'd guess your problem lays not in 'MyObject' but in 'OtherObject'. Did you try taking out marshalling/unmarshalling just that attribute?
-
caw over 11 yearsThanks!
savedInstanceState.setClassLoader(getClass().getClassLoader());
looks good but I'm afraid I can only call this for one class. What if I have to put objects from more than one class into thatBundle
? What class to call that for? -
caw over 11 yearsI've replaced all
readXXXArray()
withcreateXXXArray()
now. Moreover, I had been reading arrays of parcelable objects like this:obj[] = (Obj[]) in.readParcelableArray(Obj.class.getClassLoader())
Now I am reading them like this:obj[] = in.createTypedArray(...)
With these two changes, it seems like the problem is solved. Could this have been the cause for those problems? -
caw over 11 yearsFurthermore, with some changes, it seems as if the problem is resolved now: stackoverflow.com/questions/13997550/… Could you imagine this way of reading the objects was what caused the problems?
-
alexanderblom over 11 yearsThis will work for all your classes as getClass().getClassLoader() will retrive the APK classloader (which can load all of your classes). When reading using readXXXArray(), what arrays did you pass in? How do you know that they are of the correct length? createTypedArray() and others will not write/read type information which means that classloaders are not really involved.
-
caw over 11 yearsI didn't know that
readXXXArray()
is not the adequate function as the documentation is very poor on this topic. So basically, one should always usecreateXXXArray()
, unless an array of a specific length is needed, right? AndwriteTypedArray()
should be preferred towriteParcelableArray()
as well, shouldn't it? As my class implementsParcelable
, I can use both. -
caw over 11 yearsBy the way, do I need to set the class loader for the
Bundle
inonSaveInstanceState()
only (when saving) or inonCreate()
(when restoring) as well? -
alexanderblom over 11 yearsThe documentation is awful here, I looked at the source instead. Yeah, readXXXArray() is only useful if you have a fixed size array. I think writeTypedArray() is a bit cleaner yes, I'm not really sure what their intention with these different calls were though. You should only need to set it when restoring, when saving you already have the classes loaded so their classloader is known.
-
Julian Sievers almost 10 yearsI tried
savedInstanceState.setClassLoader(getClass().getClassLoader());
inonCreate
of my fragment but this didn't change anything. Do you have any ideas why this doesn't work?