Android混淆

1,369 阅读2分钟

Android混淆

Android Studio默认集成Java语言的ProGuard作为压缩、优化和混淆工具,配合Gradle构建工具使用很方便。本文主要介绍混淆的使用

  • ProGuard功能
    • 优化(Optimization):默认开启,在字节码级别执行优化,让app运行更快
    • 压缩(Shrinking):默认开启,用于减小app大小,移除未被使用的类和成员,在优化之后执行
    • 混淆(Obfuscation):默认开启,增加反编译难度,类和类成员会被随机命名(keep保护的除外)
      • 混淆后的日志需用mapping文件解混淆后才可定位问题

启动混淆

项目app目录下的build.gradle.kts文件增加配置即可开启,然后在proguard-rules.pro文件中添加混淆规则

android {
    ...
    buildTypes {
        release {
            // 开启混淆,代码压缩
            isMinifyEnabled = true

            // 移除无用资源
            isShrinkResources = true
            proguardFiles(
                getDefaultProguardFile("proguard-android-optimize.txt"),
                "proguard-rules.pro"
            )
        }
    }
    ...
}
...

混淆规则

注意事项

  • 实体类不混淆:与服务器交互的实体类或需解析的json数据实体类
  • 三方库混淆规则不可遗漏
  • jni方法不可混淆,需与native保持一致,js调用不可混淆
  • 反射类不可混淆:枚举类方法会被反射调用,避免混淆
  • AndroidManifest中的类声明的类不可混淆
  • 自定义组件不可混淆
  • 序列化类不混淆,Parcelable$Creator混淆会抛异常BadParcelableException异常

混淆

# ---------- 基本配置(基本不会改动) ----------
# 代码混淆压缩比,0~7,默认5
-optimizationpasses 5
# 混淆时不使用大小写混合,混淆后的类名为小写
-dontusemixedcaseclassnames
# 指定不去忽略非公共库的类
-dontskipnonpubliclibraryclasses
# 指定不忽略包可见的库类成员(字段和方法),默认解析时会跳过这些类成员
-dontskipnonpubliclibraryclassmembers
# 去掉ProGuard预校验步骤(Android不需要),可加快混淆速度
-dontpreverify
# 包含verbose,混淆后会生成映射文件,printmapping指定混淆文件名,文件中包含类名与混淆后类名映射关系,可作为解混淆参考
-verbose
-printmapping proguardMapping.txt
# 混淆算法,参数为过滤器,谷歌推荐,无需修改
-optimizations !code/simplification/cast,!field/*,!class/merging/*

# Annotation不混淆,InnerClasses视情况添加
-keepattributes *Annotation*,InnerClasses
# 泛型不混淆
-keepattributes Signature
# 抛出异常时,保留代码行号
-keepattributes SourceFile,LineNumberTable

# ---------- 默认不混淆类 ----------
# 保留本地native方法不被混淆
-keepclasseswithmembernames class * {
    native <methods>;
}

# 保留继承自Activity、Application等的子类
-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 * extends android.view.View
-keep public class com.android.vending.licensing.ILicensingService
-keep class android.support.** {*;}

# 保留Activity中参数是View的方法,避免layout中onClick指定方法无法调用
-keepclassmembers class * extends android.app.Activity{
    public void *(android.view.View);
}

# 确保枚举类不被混淆
-keepclassmembers enum * {
    public static **[] values();
    public static ** valueOf(java.lang.String);
}

# 确保自定义View不被混淆
-keep public class * extends android.view.View{
    *** get*();
    void set*(***);
    public <init>(android.content.Context);
    public <init>(android.content.Context, android.util.AttributeSet);
    public <init>(android.content.Context, android.util.AttributeSet, int);
}

# 确保序列化类不被混淆
-keep class * implements android.os.Parcelable {
    public static final android.os.Parcelable$Creator *;
}
-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();
}

# 确保R(资源)的所有类及其方法不被混淆
-keep class **.R$* {
    *;
}

# 确保带有回调函数onXxxEvent的不被混淆
-keepclassmembers class * {
    void *(**On*Event);
}

# WebView混淆
-keepclassmembers class fqcn.of.javascript.interface.for.Webview {
    public *;
}
-keepclassmembers class * extends android.webkit.WebViewClient {
    public void *(android.webkit.WebView, java.lang.String, android.graphics.Bitmap);
    public boolean *(android.webkit.WebView, java.lang.String);
}
-keepclassmembers class * extends android.webkit.WebViewClient {
    public void *(android.webkit.WebView, jav.lang.String);
}

# ---------- 三方库混淆 ----------
# 三方库介绍文档中一般会有混淆规则,拷贝过来即可

# ---------- 定制化混淆 ----------
# 一些实体类需要与服务器进行交互及各种json交互,需保留
# 保留实体类不被混淆,确保实体类放入同一个包路径下,避免遗漏
-keep public class com.dcxing.data.** {
    public void set*(***);
    public *** get*();
    public *** is*();
}

# 保留某个包下所有类不混淆
-keep class com.dcxing.data.** {*;}

# 保留某个类不混淆
-keep public class com.dcxing.Test {*;}

# 保留某个类的内部类不混淆
-keep class com.dcxing.Test$* {*;}

# 保留某个子类不混淆
-keep public class * extends class com.dcxing.AbsText {*;}

# 保留某个接口实现类不混淆
-keep class * implements class com.dcxing.IText {*;}

# 保留类构造方法和特定方法不混淆
-keepclassmembers class com.dcxing.Test {
    public <init>();
    public void test(java.lang:String);
}

混淆属性介绍

  • keepnames:保留类和类中成员不混淆,成员未被引用时会被移除
  • keepclassmembers:只保留类中成员不混淆或不被移除
  • keepclassmembernames:只保留类中成员不混淆,但成员未被引用时会被移除
  • keepclasseswithmembers:保留类和类中成员不混淆或不被移除
  • keepclasseswithmembernames:保留类和类中成员不混淆,但成员未被引用时会被移除