I have an android application which is showing strange behavior. I have a method which deserializes object from file. Below is the code I'm using:
public static Object readData(Context context, String fileName)
{
synchronized (context) {
ObjectInputStream input = null;
Object object = null;
if (fileExists(context, fileName)) {
try {
input = new ObjectInputStream(context.openFileInput(fileName));
object = input.readObject();
Log.v(Constant.TAG, "Writable object has been loaded from file "+fileName);
} catch (IOException e) {
Log.e(Constant.TAG, e.getMessage(), e);
} catch (ClassNotFoundException e) {
Log.e(Constant.TAG, e.getMessage(), e);
} finally {
try {
if (input != null)
input.close();
} catch (IOException e) {
Log.e(Constant.TAG, e.getMessage(), e);
}
}
}
return object;
}
}
Normally it works well, but when someone minimize my application and after sometime reopens, it crashes. From crash report I found that it's throwing IllegalArgumentException
in below line from above code
object = input.readObject();
I gone through the documentation of ObjectInputStream.readObject
but it doesn't state circumstances under which it can throw IllegalArgumentException
.
It is only happening when user is bringing app from background. It works perfectly well when app starts (by start I mean when app was not running, not even in background).
PS: there are some crash reports which show ClassCastException
on the same line, which is even stranger as I'm not casting, only reading into an Object
.
Update
Stack trace
java.lang.RuntimeException:
at android.app.ActivityThread.performLaunchActivity (ActivityThread.java:2423)
at android.app.ActivityThread.handleLaunchActivity (ActivityThread.java:2483)
at android.app.ActivityThread.access$900 (ActivityThread.java:153)
at android.app.ActivityThread$H.handleMessage (ActivityThread.java:1349)
at android.os.Handler.dispatchMessage (Handler.java:102)
at android.os.Looper.loop (Looper.java:148)
at android.app.ActivityThread.main (ActivityThread.java:5441)
at java.lang.reflect.Method.invoke (Native Method)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run (ZygoteInit.java:738)
at com.android.internal.os.ZygoteInit.main (ZygoteInit.java:628)
Caused by: java.lang.IllegalArgumentException:
at java.lang.reflect.Field.set (Native Method)
at java.io.ObjectInputStream.readFieldValues (ObjectInputStream.java:1127)
at java.io.ObjectInputStream.defaultReadObject (ObjectInputStream.java:454)
at java.io.ObjectInputStream.readObjectForClass (ObjectInputStream.java:1345)
at java.io.ObjectInputStream.readHierarchy (ObjectInputStream.java:1242)
at java.io.ObjectInputStream.readNewObject (ObjectInputStream.java:1835)
at java.io.ObjectInputStream.readNonPrimitiveContent (ObjectInputStream.java:761)
at java.io.ObjectInputStream.readObject (ObjectInputStream.java:1983)
at java.io.ObjectInputStream.readObject (ObjectInputStream.java:1940)
at com.pixyfisocial.pixyfi.util.IOUtil.readData (IOUtil.java:245)
at com.pixyfisocial.Login.getLoggedInUserInfoFromCache (Login.java:313)
at com.pixyfisocial.Login.startApp (Login.java:124)
at com.pixyfisocial.Login.onCreate (Login.java:98)
at android.app.Activity.performCreate (Activity.java:6303)
at android.app.Instrumentation.callActivityOnCreate (Instrumentation.java:1108)
at android.app.ActivityThread.performLaunchActivity (ActivityThread.java:2376)
A cursory examination of the sourcecode indicates that
IllegalArgumentException
is typically thrown inObjectInputStream
when there is a mismatch between the serialized object representation and what the reading class is expecting. For example, you may have incompatible customreadObject
andwriteObject
methods. Or you may have made a binary-incompatible1 change to the object representation without changing hard-codedserialVersionUID
numbers or implementing custom methods to deal with this2.There will be more clues in your stacktraces ... and in the source code of
ObjectInputStream
.The
ClassCastException
s could be another manifestation of this.1 - Changes leading to semantic incompatibility are a different matter. They won't lead to an
IllegalArgumentException
, but you should do something about them anyway.2 - If you want to cope with incompatibilities, then you probably don't want to change the
serialVerionUID
. If you do, then you will need to do "clever things" with classloading and multiple versions of the class. But the flip-side is that if your code needs to cope with multiple representations that have the sameserialVersionUID
, the representation version must be deducible from the representation itself. This requires planning.FOLLOW-UP
I took your stacktrace and tried to match it against the Android source code available at https://android.googlesource.com
The line numbers don't match exactly, but I thinkthe problem is happening in the method below. Specifically the line I have labeled "HERE". According to the javadoc for Field.set:
One of these three things is happening. It is not possible to say which one ... unless you provide a full working MCVE (that someone can run on an Android emulator!) ... but the indications point to you having (somehow) broken the serialization compatibility rules.
Note, since the line numbers didn't match, I cannot say with certainty that the Android you are using matches this following. If you want to be sure, you need to search the history in the GIT repo to find a matching version .... or look in the vendor specific source code bundle / repo for your device.