Подтвердить что ты не робот

ProGuard и Gson на Android (ClassCastException)

У меня есть ад с Гсоном и Прогуаром. У меня есть простой объект, и когда я разбираю тойсон, сохраняю в sqllite и читаю обратно из базы данных, чтобы загрузить json обратно в мой объект, я получаю java.lang.classcastexception. Если я не использую ProGuard, все работает отлично.

Я проверил, что строка json, отправленная и полученная из базы данных, является такой же. Исключение не выбрасывается, когда оно преобразуется из json, но, скорее, когда я пытаюсь получить доступ к объекту.

Вот мой простой объект:

public class ScanLog extends ArrayList<SingleFrame>
{
     private static final long serialVersionUID = 1L;

     public ScanLog()
     {

     }
}

public final class SingleFrame 
{
    public int Position;
    public int Time;
    public Map<Integer,String> MainDataMap;
    public Map<Integer,String> DataMap;

    public SingleFrame(int position, int time, 
                    Map<Integer,String> mainDataMap, Map<Integer,String> dataMap)
    {
        this.Position = position;
        this.Time = time;
        this.MainDataMap = mainDataMap;
        this.DataMap = dataMap;
    }

}

Все другие аспекты моего приложения прекрасны, но что-то с proguard вызывает это. Я пробовал всевозможные команды -keep в proguard.cfg, но я не уверен, что Im делает правильно.

ИЗМЕНИТЬ - ДОБАВИТЬ PROGUARD.CFG

-dontusemixedcaseclassnames
-dontskipnonpubliclibraryclasses
-dontpreverify
-verbose
-dontshrink
-dontoptimize

-keep public class * extends android.app.Activity
-keep public class * extends android.app.Application
-keep public class * extends android.app.Service
-keep public class * extends android.content.BroadcastReceiver
-keep public class * extends android.content.ContentProvider
-keep public class * extends android.app.backup.BackupAgentHelper
-keep public class * extends android.preference.Preference
-keep public class com.android.vending.licensing.ILicensingService

#keep all classes that might be used in XML layouts
-keep public class * extends android.view.View
-keep public class * extends android.app.Fragment
-keep public class * extends android.support.v4.Fragment

#keep all classes
-keep public class *{
public protected *;
}

#keep all public and protected methods that could be used by java reflection
-keepclassmembernames class * {
    public protected <methods>;
}


-keepclasseswithmembernames class * {
    native <methods>;
}

-keep public class org.scanner.scanlog.SingleFrame


-keepclassmembers class org.scanner.scanlog.ScanLog { 
        private <fields>; 
        public <fields>; 
}

-keepclassmembers class org.scanner.scanlog.SingleFrame { 
        private <fields>; 
        public <fields>; 
}

-keepclasseswithmembernames class * {
    public <init>(android.content.Context, android.util.AttributeSet);
}

-keepclasseswithmembernames class * {
    public <init>(android.content.Context, android.util.AttributeSet, int);
}

-keepclassmembers enum * {
    public static **[] values();
    public static ** valueOf(java.lang.String);
}

-keep class * implements android.os.Parcelable {
  public static final android.os.Parcelable$Creator *;
}

-dontwarn **CompatHoneycomb
-dontwarn org.htmlcleaner.*
#-keep class android.support.v4.** { *; }

EDIT - Хорошо, что я получил ACRA, успешно настроенный в моем приложении, довольно потрясающая функция! Вот трассировка стека:

java.lang.ClassCastException: java.lang.Object
    at org.scanner.activity.ReaderMainActivity.AdvanceScanLog(SourceFile:1499)
    at org.scanner.activity.r.onProgressChanged(SourceFile:271)
    at android.widget.SeekBar.onProgressRefresh(SeekBar.java:89)
    at android.widget.ProgressBar.doRefreshProgress(ProgressBar.java:507)
    at android.widget.ProgressBar.refreshProgress(ProgressBar.java:516)
    at android.widget.ProgressBar.setProgress(ProgressBar.java:565)
    at android.widget.AbsSeekBar.trackTouchEvent(AbsSeekBar.java:337)
    at android.widget.AbsSeekBar.onTouchEvent(AbsSeekBar.java:292)
    at android.view.View.dispatchTouchEvent(View.java:3932)
    at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:906)
    at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:906)
    at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:906)
    at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:906)
    at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:906)
    at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:906)
    at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:906)
    at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:906)
    at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:906)
    at com.android.internal.policy.impl.PhoneWindow$DecorView.superDispatchTouchEvent(PhoneWindow.java:1784)
    at com.android.internal.policy.impl.PhoneWindow.superDispatchTouchEvent(PhoneWindow.java:1157)
    at android.app.Activity.dispatchTouchEvent(Activity.java:2181)
    at com.android.internal.policy.impl.PhoneWindow$DecorView.dispatchTouchEvent(PhoneWindow.java:1759)
    at android.view.ViewRoot.deliverPointerEvent(ViewRoot.java:2336)
    at android.view.ViewRoot.handleMessage(ViewRoot.java:1976)
    at android.os.Handler.dispatchMessage(Handler.java:99)
    at android.os.Looper.loop(Looper.java:143)
    at android.app.ActivityThread.main(ActivityThread.java:4263)
    at java.lang.reflect.Method.invokeNative(Native Method)
    at java.lang.reflect.Method.invoke(Method.java:507)
    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:839)
    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:597)
    at dalvik.system.NativeStart.main(Native Method)
4b9b3361

Ответ 2

Эти настройки в конфигурации работали для меня в одном из моих приложений:

# Add the gson class
-keep public class com.google.gson

# Add any classes the interact with gson
-keep class com.someapp.android.models.ChatModel { *; }
-keep class com.someapp.android.models.FeedModel { *; }

# Add the path to the jar
-libraryjars /Users/someuser/Documents/workspace/someapp/lib/gson-1.7.1.jar

Надеюсь, это поможет вам.

Ответ 3

Применение изменений, найденных в Android-приложении в проекте Gson, работало для меня

Необходимые строки:

-keepattributes Signature
-keep class sun.misc.Unsafe { *; }
# and keeping the classes that will be serialized/deserialized

Ответ 4

Я знаю, что исходный вопрос был решен с помощью другого подхода, но у меня была очень похожая проблема с использованием flexjson и Proguard на Android, и я решил это, если кто-то наткнулся на нее сам.

При переходе от JSON к моему объекту value, который включал некоторые ArrayLists, я получал бы такое же ClassCastException. Я получил его, чтобы работать, в основном, с запутанием, но отключая все части обфускации (-поддерживайте все, -keepclassmembers все и -keepattributes все), а затем работая назад, позволяя вещи немного за раз.

Результат; сохраняя всю библиотеку flexjson:

-keep class flexjson**
--keepclassmembers class flexjson** {
   *;
}

и сохранения атрибута Signature and Annotation:

-keepattributes Signature, *Annotation*

Я смог использовать библиотеку flexjson без инцидентов после этого в расширенной версии выпуска моего приложения.

Ответ 5

Я получал ошибки для классов модели с proguard. Если вы посмотрите на

GSON Proguard вы найдете строку

# Application classes that will be serialized/deserialized over Gson
-keep class com.google.gson.examples.android.model.** { *; }

замените com.google.gson.examples.android.model. на ваш пакет model, также в моем случае я заменил его на -keep class com.consumer.myProject.model.** { *; }

остальное я скопировал как таковой

Ответ 6

Это похоже на то, что вы держите все от своего класса (поля, методы и сам класс). Но чтобы убедиться, что вы можете добавить -printseeds outputfile.txt в файл proguard.cfg, чтобы убедиться, что proguard действительно сохраняет все, что вам нужно, после того, как обфускация будет выполнена.

Кстати, вы можете подумать о добавлении чего-то вроде ACRA или Android Remote stacktrace, который позволяет вам проверять стек стеков в встроенном приложении.

Ответ 7

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

Я в целом более счастлив, хотя это стоило мне более 12 часов, пытаясь понять это. По-видимому, PROGUARD и gson не должны слишком сильно любить друг друга?

Поскольку HUGELY добавил преимущество, чтобы не использовать GSON, я заметил, что, выгружая библиотеку GSON, размер моего приложения сокращается вдвое. Мое приложение было 577kb и теперь только 260kb после удаления gson lib.