一.简介:
平时在做项目中,版本发布之前往往要对自己的代码做一下混淆,以免别人反编译自己的代码。代码混淆其实并不复杂但代码混淆后有很多需要注意的地方,比如自己的数据结构需继承serializable接口的bean类如果忘记继承这个接口了,在没有混淆前数据可以解析出来,但混淆后数据就解析不出来了因为这样的类是不能被混淆的。
二.实现:
1.首先在gradle里配置:
buildTypes { debug { minifyEnabled true proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' } release { minifyEnabled true proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' } }minifyEnable为true是开启混淆,为false是关闭混淆。
2.在gradle同级目录下有个proguard-rules.pro文件就是上边配置时写的那个文件名:
#指定代码的压缩级别 -optimizationpasses 5 #包明不混合大小写 -dontusemixedcaseclassnames #不去忽略非公共的库类 -dontskipnonpubliclibraryclasses #优化 不优化输入的类文件 -dontoptimize #预校验 -dontpreverify #混淆时是否记录日志 -verbose # 混淆时所采用的算法 -optimizations !code/simplification/arithmetic,!field/*,!class/merging/* #保护注解 -keepattributes *Annotation* # 保持哪些类不被混淆 -keep public class * extends android.app.Fragment -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 #如果有引用v4包可以添加下面这行 -keep public class * extends android.support.v4.app.Fragment #忽略警告 -ignorewarning #####################记录生成的日志数据,gradle build时在本项目根目录输出################ #apk 包内所有 class 的内部结构 -dump class_files.txt #未混淆的类和成员 -printseeds seeds.txt #列出从 apk 中删除的代码 -printusage unused.txt #混淆前后的映射 -printmapping mapping.txt #####################记录生成的日志数据,gradle build时 在本项目根目录输出-end################ #如果引用了v4或者v7包 -dontwarn android.support.** ############<span></span>混淆保护自己项目的部分代码以及引用的第三方jar包library-end################## -keep public class * extends android.view.View { public <init>(android.content.Context); public <init>(android.content.Context, android.util.AttributeSet); public <init>(android.content.Context, android.util.AttributeSet, int); public void set*(...); } #保持 native 方法不被混淆 -keepclasseswithmembernames class * { native <methods>; } #保持自定义控件类不被混淆 -keepclasseswithmembers class * { public <init>(android.content.Context, android.util.AttributeSet); } #保持自定义控件类不被混淆 -keepclasseswithmembers class * { public <init>(android.content.Context, android.util.AttributeSet, int); } #保持自定义控件类不被混淆 -keepclassmembers class * extends android.app.Activity { public void *(android.view.View); } #保持 Parcelable 不被混淆 -keep class * implements android.os.Parcelable { public static final android.os.Parcelable$Creator *; } #保持 Serializable 不被混淆 #-keepnames class * implements java.io.Serializable #保持 Serializable 不被混淆并且enum 类也不被混淆 -keepclassmembers class * implements java.io.Serializable { static final long serialVersionUID; private static final java.io.ObjectStreamField[] serialPersistentFields; !static !transient <fields>; !private <methods>; private void writeObject(java.io.ObjectOutputStream); private void readObject(java.io.ObjectInputStream); java.lang.Object writeReplace(); java.lang.Object readResolve(); } #保持枚举 enum 类不被混淆 如果混淆报错,建议直接使用上面的 -keepclassmembers class * implements java.io.Serializable即可 #-keepclassmembers enum * { # public static **[] values(); # public static ** valueOf(java.lang.String); #} -keepclassmembers class * { public void *ButtonClicked(android.view.View); } #不混淆资源类 -keepclassmembers class **.R$* { public static <fields>; } #避免混淆泛型 如果混淆报错建议关掉 #–keepattributes Signature #移除log 测试了下没有用还是建议自己定义一个开关控制是否输出日志 -assumenosideeffects class android.util.Log { public static boolean isLoggable(java.lang.String, int); public static int v(...); public static int i(...); public static int w(...); public static int d(...); public static int e(...); } #gson #-libraryjars libs/gson-2.2.2.jar -keepattributes Signature # Gson specific classes -keep class sun.misc.Unsafe { *; } -keep class com.google.gson.stream.** { *; } #-keep public class * extends android.support.v4.widget.SwipeRefreshLayout{ # public *; # } # -keep public class * extends java.lang.String{ # public *; # } #-keep class packagename.** {*;} # Application classes that will be serialized/deserialized over Gson -keep class com.google.gson.examples.android.model.** { *; } # webview + js -keepattributes *JavascriptInterface* # keep 使用 webview 的类 #-keepclassmembers class com.veidy.activity.WebViewActivity { # public *; #} # keep 使用 webview 的类的所有的内部类 #-keepclassmembers class com.veidy.activity.WebViewActivity$*{ # *; #} # keep 使用 webview 的类 -keepclassmembers class 自己webview类的文件路径 { public *; } #友盟添加 -keepclassmembers class * { public <init>(org.json.JSONObject); } -keepclassmembers enum * { public static **[] values(); public static ** valueOf(java.lang.String); } # keep 使用 webview 的类 -keepclassmembers class com.example.blke.base.BaseWebActivity$* { *; } # keep 使用 webview 的类的所有的内部类 -keepclassmembers class com.example.blke.base.AWebActivity$*{ *; } #保留js交互的代码 -keep class js代码类路径{*;}3.注意事项:
(1).加入第三方的东西时要看第三方需不需要添加混淆,需要的话直接添加在里面就可以。
(2).webview不能被混淆,js类不能被混淆用-keep保证其不被混淆。
(3).混淆前后一定要全面测试一下会不会出问题。