Сериализация/Deserialization & Proguard

В одном из моих приложений у меня возникла проблема с одним из моих классов Serialized, когда я пытаюсь обновить APK.

Действительно, были проблемы, связанные с объектами, сохраненными с предыдущей версией apk и новой версией apk.

В последнем APK (в производстве на Android Market) я забыл сконфигурировать свой proguard.cfg для класса Serializable (и, следовательно, его член статический конечный long serialVersionUID)...

Поэтому, когда я пытаюсь в своем новом APK перезагрузить этот ранее сохраненный класс Serializable, у меня есть проблема InvalidClassException в DDMS StackTrace:

04-24 18:17:40.120: W/System.err(1204): java.io.InvalidClassException: cu; Incompatible class (SUID): cu: static final long serialVersionUID =6593847879518920343L; but expected cu: static final long serialVersionUID =0L;
04-24 18:17:40.125: W/System.err(1204):     at java.io.ObjectInputStream.verifyAndInit(ObjectInputStream.java:2380)
04-24 18:17:40.125: W/System.err(1204):     at java.io.ObjectInputStream.readNewClassDesc(ObjectInputStream.java:1662)
04-24 18:17:40.125: W/System.err(1204):     at java.io.ObjectInputStream.readClassDesc(ObjectInputStream.java:683)
04-24 18:17:40.125: W/System.err(1204):     at java.io.ObjectInputStream.readNewObject(ObjectInputStream.java:1803)
04-24 18:17:40.125: W/System.err(1204):     at java.io.ObjectInputStream.readNonPrimitiveContent(ObjectInputStream.java:787)
04-24 18:17:40.125: W/System.err(1204):     at java.io.ObjectInputStream.readObject(ObjectInputStream.java:2003)
04-24 18:17:40.125: W/System.err(1204):     at java.io.ObjectInputStream.readObject(ObjectInputStream.java:1960)

Я знаю, что это была проблема обфускации с объектами Serializable и их serialVersionUID...

После прочтения Proguard и Serialized Java Objects, которые явно раскрывают мою проблему, я не могу решить свою проблему...

В моем следующем APK я добавлю это в свой proguard.cfg:

-keepnames class * implements java.io.Serializable

-keepclassmembers class * implements java.io.Serializable {
    static final long serialVersionUID;
    private static final java.io.ObjectStreamField[] serialPersistentFields;
    !static !transient ;
    private void writeObject(java.io.ObjectOutputStream);
    private void readObject(java.io.ObjectInputStream);
    java.lang.Object writeReplace();
    java.lang.Object readResolve();
}

чтобы избежать проблемы для следующих обновлений, мне действительно нужно вернуть эти старые объекты...

Я пытаюсь изменить serialVersionUID с помощью 6593847879518920343L или 0L, без успеха...

Любая идея?

Заранее благодарим за ваши ответы!

Ответ 1

Вы можете попробовать следующее:

  • Вычислить serialVersionUID от обфускации сериализуемых классов и добавить их в текущий исходный код.
  • Обфускайте новый код, сохраняя идентификаторы serialVersionUID, но также удостоверяясь, что сериализуемые классы сопоставляются с более ранними обфускамированными именами (с опцией -applymapping).