混淆配置
android{
buildTypes {
release {
buildConfigField "boolean", "LOG_DEBUG", "false" //不显示log
minifyEnabled true
shrinkResources true
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
signingConfig signingConfigs.config
}
}
}
开启混淆会使编译时间变长,所以一般debug模式下不开启。
开启混淆
minifyEnabled
的值改为true
,打开混淆;
混淆文件
proguardFiles
配置混淆文件,后面可以多文件配置,例如:
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules1.pro','proguard-rules2.pro' ,...
getDefaultProguardFile('proguard-android.txt')
为android studio提供的默认混淆方案,该文件位于<sdk>/tools/proguard,同样位于该文件夹下还有一个文件proguard-android-optimize.txt
,和proguard-android.txt
的区别在于,proguard-android-optimize.txt
中开启了Proguard optimize
的选项(optimize是Proguard的一项功能),而proguard-android.txt
中没有开启optimize选项。
proguard-rules.pro
默认放置于模块根目录,是开发者自定义混淆规则的地方。
optimize
optimize过程会在Java字节码层面上进行优化,剔除方法中一些冗余的调用。帮助文档中列出了一些当前Proguard支持的优化。
对常量表达式进行计算
删除不必要的字段访问和方法调用
删除不必要的代码分支
删除不必要的比较和instanceof测试
删除未使用的代码块
合并相同的代码块
减少变量的分配
删除只赋值但没有使用的成员变量,以及未使用的方法参数
内联常量字段,方法参数和返回值
内联很短的方法和只被调用一次的方法
简化尾递归调用
合并类和接口
尽可能的将方法修饰为private,static和final
尽可能的将类修饰为static和final
移除只被一个类实现的接口
其他200多项细节优化,比如用 <<1 代替 *2 运算
删除Log的相关代码(可选)
混淆规则
[保持命令] [类] {
[成员]
}
保持命令
命令 | 作用 |
---|---|
-keep | 防止类和成员被移除或者被重命名 |
-keepnames | 防止类和成员被重命名 |
-keepclassmembers | 防止成员被移除或者被重命名 |
-keepclassmembernames | 防止成员被重命名 |
-keepclasseswithmembers | 防止拥有该成员的类和成员被移除或者被重命名 |
-keepclasseswithmembernames | 防止拥有该成员的类和成员被重命名 |
-keepclassmembers interface com.xbxz.bms.common.global.MenuConstant{ *;}
表示:类的成员不做混淆,类名可以混淆
类
代表类相关的限定条件,它将最终定位到某些符合该限定条件的类。它的内容可以使用:
- 具体的类
- 访问修饰符(public、protected、private)
- 通配符*,匹配任意长度字符,但不含包名分隔符(.)
- 通配符**,匹配任意长度字符,并且包含包名分隔符(.)
- extends,即可以指定类的基类
- implement,匹配实现了某接口的类
- $,内部类
-keep public class * extends android.app.Activity
表示:继承自Activity的所有public类的类名不做混淆,内部属性,方法仍会被混淆
成员
代表类成员相关的限定条件,它将最终定位到某些符合该限定条件的类成员。它的内容可以使用:
- <init> 匹配所有构造器
-keep public class com.biaobiao.test.** extends Android.app.Activity {
<init>
}
表示:com.biaobiao.test包下继承自Activity的所有类的构造函数不做混淆
-keepclassmembers class com.xbxz.bms.common.application.BaseApplication {
public <init>();
public <init>(android.content.Context);
}
表示:类BaseApplication 的无参构造函数不做混淆,同样也可以指定具体的带参构造函数
- <fields> 匹配所有域
-keepclassmembers class **$WhenMappings {
<fields>;
public <fields>;
public static final <fields>;
}
表示:该类下的所有成员不做混淆,包括静态,非静态,同样可以指定修饰符来做限定
- <methods> 匹配所有方法
- 通配符*,匹配任意长度字符,但不含包名分隔符(.)
-keep class cn.jpush.** { *; }
表示:整个类不做混淆
- 通配符**,匹配任意长度字符,并且包含包名分隔符(.)
- 通配符***,匹配任意参数类型
- …,匹配任意长度的任意类型参数。比如void test(…)就能匹配任意 void test(String a) 或者是 void test(int a, String b) 这些方法。
-keepclassmembers class **$** extends com.chad.library.adapter.base.viewholder.BaseViewHolder {
<init>(...);
}
通过 ... 形式匹配所有的构造,不做混淆
**$**表示所有的内部类
- 访问修饰符(public、protected、private)
混淆设置参数
设置 | 介绍 |
---|---|
-optimizationpasses 4 | 代码混淆的压缩比例,值介于0-7 |
-dontusemixedcaseclassnames | 混淆后类型都为小写 |
-dontskipnonpubliclibraryclasses | 不去忽略非公共的库类 |
-dontoptimize | 不优化输入的类文件 |
-dontpreverify | 不做预校验的操作 |
-ignorewarnings | 忽略警告 |
-dontwarn kotlin.** | 不做警告(多用于第三方库) |
-verbose | 混淆时是否记录日志 |
-keepattributes | Annotation 保护注解 |
-printmapping proguardMapping.txt | 生成原类名和混淆后的类名的映射文件 |
-optimizations | !code/simplification/cast,!field/ ,!class/merging/ 指定混淆是采用的算法 |