Android上的ProGuard和Gson(ClassCastException)

ProGuard and Gson on Android (ClassCastException)

我和GSON和Proguard玩得很开心。我有一个简单的对象,当我分析to json、保存到sqllite并从数据库中读取时,为了将json加载回我的对象,我得到了java.lang.ClassCastException。如果我不使用Proguard,一切正常。

我已经验证了发送到数据库和从数据库中获取的JSON字符串是相同的。当它从JSON转换时,不会引发异常,而是在我尝试访问对象时引发异常。

这是我的简单对象:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
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的某些功能导致了这种情况的发生……我在proguard.cfg中尝试了各种-keep命令,但我不确定我所做的是否正确。

编辑-添加proguard.cfg

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
-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.** { *; }

编辑-好吧,我在我的应用程序中成功设置了ACRA,非常棒的功能!下面是堆栈跟踪:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
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)


有关推荐的Proguard配置文件的最新版本,请参阅GSON提供的Android Proguard示例:https://github.com/google/gson/blob/master/examples/android-proguard-example/proguard.cfg


配置中的这些设置在我的一个应用程序中对我有效:

1
2
3
4
5
6
7
8
9
# 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

希望这能帮到你。


将Android示例中的更改应用于GSON项目对我有效

需要的线路是:

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


我知道最初的问题是通过采用不同的方法解决的,但是我在Android上使用flexjson和proguard时遇到了一个非常类似的问题,我已经解决了,以防有人自己碰到它。

当从JSON转换回包含一些数组列表的值对象时,我会得到相同的ClassCastException。我基本上通过启用模糊来实现它,但是关闭模糊的所有部分(保留所有内容,-keepclassmembers所有内容,并-keepattributes所有内容),然后通过一次启用一点来向后工作。

结果;保留整个flexjson库:

1
2
3
4
-keep class flexjson**
--keepclassmembers class flexjson** {
   *;
}

保留签名和注释属性:

1
-keepattributes Signature, *Annotation*

之后,我可以在我的应用程序的一个预编程版本中使用flexjson库,而不会发生任何意外。


如果你看的话,我在使用proguard的模型类中出错了

GSON Proguard你会找到一条线

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

用你的model包替换com.google.gson.examples.android.model.,我也用-keep class com.consumer.myProject.model.** { *; }包替换。

剩下的我照搬了


因此,我最终放弃了GSON库,并没有使用GSON将我的对象转换为JSON,而是在我的应用程序中创建了一个自定义类来序列化和反序列化对象,并以这种方式存储数据。

总的来说,我更高兴,尽管这让我花了12个多小时的时间去弄清楚。显然,Proguard和GSON不能太喜欢对方?

作为一个不必使用GSON的巨大好处,我注意到,通过删除GSON库,我的应用程序大小减少了一半。我的应用程序是577kb,在删除gson lib之后现在只有260kb。


它确实看起来像是将所有内容都与类(字段、方法和类本身)保持在一起。但为了确保您可以将-printseeds outputfile.txt添加到proguard.cfg文件中,以验证proguard在模糊处理完成后是否真正保留了所需的一切。

顺便说一句,您可能会考虑添加一些类似ACRA或Android远程stacktrace的东西,这些东西允许您在构建的应用程序上检查stacktrace。