Android Proguard混淆打包经验总结
mlc0202
8年前
<p>作为一名 Android 开发,应该了解并尝试给自己的项目进行Proguard混淆打包。项目经过Proguard混淆打包后,会发现apk包体积会变小,也就是混淆可以使得apk瘦身;并且反编译apk的时候会发现, 项目中的源码都被处理过,进一步保障了apk的安全;这就是我所理解的Proguard混淆打包的两个优点。不过,想要真正给自己的项目进行Proguard混淆打包,可不是一件容易的事情,真正尝试去做了,才会发现有好多问题需要去解决,毕竟,混淆打包是针对特定的项目,每个项目需要混淆的代码都是有区别的。不过,所有apk混淆打包,也有一些通用的规则处理,像这些规则,就可以自己记录下来,这样其他项目混淆打包的时候就可以复制粘贴使用混淆代码了。好了,进入正题,如何给自己的项目量身定制一套Proguard混淆代码了?</p> <p>关于Proguard混淆,给APP瘦身,Google官方也给出了文档给了大致解释,大家有兴趣可以看看,纯英文(够呛),不过怎么在Android Studio配置Proguard混淆,还是可以看懂的:</p> <p>我们可以从这些方面对apk瘦身:</p> <p>1)冗余的代码,比如多余的jar包代码;</p> <p>2)未使用的静态代码;</p> <p>3)资源代码的冗余;</p> <p>4)native code</p> <p>5)图片资源的优化和压缩</p> <p>ProGuard混淆打包,解决的问题主要针对第一点,一般项目进行ProGuard混淆打包后,apk的体积会减小200-500KB左右。</p> <h2><strong>一、在Android Studio中配置Proguard混淆</strong></h2> <pre> <code class="language-java">buildTypes { release { //混淆 minifyEnabled true proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' } }</code></pre> <p>配置很简单,只需要把minifyEnabled的状态改成true即可,然后,当进行apk打包时,编译工具首先会读取proguar-rules.pro文件。所有混淆规则,即项目的专属混淆配置代码就写在该文件里面。</p> <h2><strong>二、编写ProGuard文件</strong></h2> <p>proguar-rules.pro文件目录位置</p> <p style="text-align:center"><img src="https://simg.open-open.com/show/26cc628bd077ca9428797afde56980bf.jpg"></p> <p>1、ProGuard混淆配置文件通用配置</p> <p>这里形容的通用配置,也就是需要添加一些不需要混淆的代码,比如Android的四大组件的相关代码,自定义View相关代码,项目中引用的第三方jar包相关代码等等;</p> <p>1)ProGuadr混淆代码属性配置代码</p> <pre> <code class="language-java">-optimizationpasses 5 -dontusemixedcaseclassnames -dontskipnonpubliclibraryclasses -dontpreverify -verbose -dontwarn -dontskipnonpubliclibraryclassmembers -ignorewarnings -optimizations !code/simplification/arithmetic,!field/*,!class/merging/* -keepattributes Exceptions,InnerClasses,Signature,Deprecated,SourceFile,LineNumberTable,*Annotation*,EnclosingMethod # 保持 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); } # 泛型与反射 -keepattributes Signature -keepattributes EnclosingMethod -keepattributes *Annotation*</code></pre> <p>以上属性基本配置代码中,配置了ProGuard混淆代码压缩比例等基本属性,配置了本地native方法、泛型与反射、注解、内部类、异常处理等相关代码的混淆配置,保证这些相关的代码不被混淆,基本上所有项目都可以通用。</p> <p>1)四大组件ProGuard配置代码</p> <pre> <code class="language-java">-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 public class * extends android.os.IInterface</code></pre> <p>2)项目中的activity、constants、model、bean等混淆配置</p> <p>比如我的项目结构如下</p> <p style="text-align:center"><img src="https://simg.open-open.com/show/249d10227d524c91402c2b4f8651e563.jpg"></p> <p>针对我的项目配置相关配置代码如下:</p> <pre> <code class="language-java">-keep class com.example.proguard.model.** {*;} -keep class com.example.proguard.activity.** {*;} -keep class com.example.proguard.adapter.** {*;} -keep class com.example.proguard.constants.**{*;} -keep class com.example.proguard.base.**{*;} -keep class com.example.proguard.model.**{*;} -keep class com.example.proguard.service.**{*;} -keep class com.example.proguard.weidgt.**{*;} -keep class com.example.proguard.utils.** {*;} -keep class com.example.proguard.network.api** {*;}</code></pre> <p>3)第三方jar包混淆配置代码</p> <p>比如我的项目中涉及到的第三方jar包如下:</p> <pre> <code class="language-java">compile fileTree(dir: 'libs', include: ['*.jar']) testCompile 'junit:junit:4.12' compile 'com.android.support:appcompat-v7:23.0.0' compile 'com.android.support:design:23.1.1' compile 'com.squareup.okhttp3:okhttp:3.2.0' compile 'com.squareup.retrofit2:retrofit:2.0.0' compile 'com.squareup.retrofit2:converter-gson:2.0.0' compile 'com.squareup.retrofit2:adapter-rxjava:2.0.0' compile 'com.jakewharton:butterknife:7.0.1' compile 'io.reactivex:rxjava:1.1.0' compile 'io.reactivex:rxandroid:1.1.0' compile 'io.reactivex:rxjava:1.0.10' compile 'com.jakewharton.rxbinding:rxbinding:0.4.0' compile 'com.jakewharton.rxbinding:rxbinding-appcompat-v7:0.4.0' compile 'com.jakewharton.rxbinding:rxbinding-design:0.4.0' compile 'com.非死book.fresco:fresco:0.8.1+' compile 'com.非死book.fresco:drawee:0.5.0+' compile 'com.非死book.fresco:fbcore:0.5.0+' compile 'com.非死book.fresco:imagepipeline:0.5.0+' compile 'com.google.code.gson:gson:2.7' compile 'com.orhanobut:logger:1.8' compile 'com.非死book.stetho:stetho:1.3.1' compile 'de.greenrobot:greendao:1.3.7' // 时间日期选择控件 里面包含nineold动画jar包 compile 'com.github.flavienlaurent.datetimepicker:library:0.0.2' //Android 6.0f权限检查 compile 'gun0912.ted:tedpermission:1.0.0'</code></pre> <p>我需要的ProGuard配置代码如下:</p> <pre> <code class="language-java">#保持第3方jar包不混淆 -keep class com.alibaba.sdk.android.** {*;} -keep class com.squareup.okhttp.** {*;} -keep class com.squareup.okhttp3.** {*;} -keep class com.squareup.retrofit2.** {*;} -keep class com.jakewharton.** {*;} -keep class io.reactivex.** {*;} -keep class rx.** {*;} -keep class com.jakewharton.rxbinding.** {*;} -keep class com.非死book.** {*;} -keep class com.butterknife.** {*;} -keep class com.google.code.gson.** {*;} -keep class com.orhanobut.** {*;} -dontwarn de.greenrobot.daogenerator.** -keep class de.greenrobot.** {*;} -keep class com.github.flavienlaurent.datetimepicker.** {*;} -keep class gun0912.ted.** {*;}</code></pre> <p>4)针对support4包的配置</p> <pre> <code class="language-java">-dontwarn android.support.** -dontwarn android.support.v4.** -keep class android.support.v4.** { *; } -keep public class * extends android.support.v4.** -keep public class * extends android.support.v4.app.Fragment</code></pre> <p>针对supportv7包的配置一样</p> <p>5)针对butterknife的配置</p> <pre> <code class="language-java">#####butterknife####### -keep class butterknife.** { *; } -dontwarn butterknife.internal.** -keep class **$$ViewBinder { *; } -dontwarn butterknife.** -keepclasseswithmembernames class * { @butterknife.* <fields>; } -keepclasseswithmembernames class * { @butterknife.* <methods>; }</code></pre> <p>6)针对gson的配置</p> <pre> <code class="language-java">-keep class com.google.gson.** {*;} #-keep class com.google.**{*;} -keep class sun.misc.Unsafe { *; } -keep class com.google.gson.stream.** { *; } -keep class com.google.gson.examples.android.model.** { *; } -keep class com.google.** { ; ; } -keepclassmembers class * implements java.io.Serializable { static final long serialVersionUID; private static final java.io.ObjectStreamField[] serialPersistentFields; private void writeObject(java.io.ObjectOutputStream); private void readObject(java.io.ObjectInputStream); java.lang.Object writeReplace(); java.lang.Object readResolve(); } -dontwarn com.google.gson.**</code></pre> <p>7)针对greenDao数据库的配置</p> <pre> <code class="language-java">-keep class de.greenrobot.dao.** {*;} #保持greenDao的方法不被混淆 用来保持生成的表名不被混淆 -keepclassmembers class * extends de.greenrobot.dao.AbstractDao { public static java.lang.String TABLENAME; } -keep class **$Properties</code></pre> <p>8)针对okhttp的配置</p> <pre> <code class="language-java"># OkHttp -dontwarn com.squareup.okhttp.** -keep class com.squareup.okhttp.** {*;} -keep interface com.squareup.okhttp.** {*;} -dontwarn okio.**</code></pre> <p>等等,针对某个jar包的具体混淆配置,大家可以去Google,这里我就列举这些了。</p> <p>至此,针对你项目的专属ProGuard配置文件就出来了,打个正式包看看apk包有没有瘦身吧。</p> <p>三、ProGuard混淆打包过程的问题</p> <p>当你配置好项目的ProGuard文件后,打包过程可能就报错了,这些错误需要我们一一去解决,这里列出几个我在打包过程遇到的几个错误及解决方法:</p> <p>1) java.lang.UnsupportedOperationException: Unknown ASTNode child: LambdaExpression</p> <p>2)如果配置好ProGuard文件后,并且打包也顺利进行,打包成功了,但是当你运行瘦身后的apk,会发现apk会出现崩溃的情况,如果出现这种情况,说明配置的ProGuard文件还有问题,有些地方的代码混淆配置还需要更改,这时候就需要我们定位到具体报错的代码位置了,去发现并定位混淆配置错误的代码。Android Studio在debug模式下运行的程序,是没有经过ProGuard混淆配置的,所以我们没法通过debug模式定位混淆出错的位置。但是我们该如何定位到代码崩溃的代码位置了,解决的方法就是在debug模式下,我们也让程序跑ProGuard文件,进行debug模式下的混淆打包。</p> <pre> <code class="language-java">buildTypes { release { //混淆 minifyEnabled true proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' } debug{ //混淆 minifyEnabled true proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' } }</code></pre> <p>这样,在debug模式下,就可以定位到程序崩溃或异常的代码了,进而可以查询到混淆配置错误的代码进行修改调整。</p> <p>好了,以上就是我在项目中配置ProGuard文件的经验总结,希望大家也能针对自己的项目做一份特制的ProGuard文件,让apk瘦身成功。</p> <p> </p> <p>来自:http://www.androidchina.net/5678.html</p> <p> </p>