Android 一些常用的混淆Proguard

一些公共的模板

#############################################
#
# 對於一些基本指令的添加
#
#############################################
# 代碼混淆壓縮比,在 0~7 之間,默認為 5,一般不做修改
-optimizationpasses 5

# 混合時不使用大小寫混合,混合後的類名為小寫
-dontusemixedcaseclassnames

# 指定不去忽略非公共庫的類
-dontskipnonpubliclibraryclasses

# 這句話能夠使我們的項目混淆後產生映射文件
# 包含有類名->混淆後類名的映射關系
-verbose

# 指定不去忽略非公共庫的類成員
-dontskipnonpubliclibraryclassmembers

# 不做預校驗,preverify 是 proguard 的四個步驟之一,Android 不需要 preverify,去掉這一步能夠加快混淆速度。
-dontpreverify

# 保留 Annotation 不混淆
-keepattributes *Annotation*,InnerClasses

# 避免混淆泛型
-keepattributes Signature

# 拋出異常時保留代碼行號
-keepattributes SourceFile,LineNumberTable

# 指定混淆是采用的算法,後面的參數是一個過濾器
# 這個過濾器是谷歌推薦的算法,一般不做更改
-optimizations !code/simplification/cast,!field/*,!class/merging/*


#############################################
#
# Android開發中一些需要保留的公共部分
#
#############################################

# 保留我們使用的四大組件,自定義的 Application 等等這些類不被混淆
# 因為這些子類都有可能被外部調用
-keep public class * extends android.app.Activity
-keep public class * extends android.app.Appliction
-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


# 保留 support 下的所有類及其內部類
-keep class android.support.** { *; }

# 保留繼承的
-keep public class * extends android.support.v4.**
-keep public class * extends android.support.v7.**
-keep public class * extends android.support.annotation.**

# 保留 R 下面的資源
-keep class **.R$* { *; }

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

# 保留在 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);
}

# 保留 Parcelable 序列化類不被混淆
-keep class * implements android.os.Parcelable {
  public static final android.os.Parcelable$Creator *;
}

# 保留 Serializable 序列化的類不被混淆
-keepnames class * implements java.io.Serializable
-keepclassmembers class * implements java.io.Serializable {
  static final long serialVersionUID;
  private static final java.io.ObjectStreamField[] serialPersistentFields;
  !static !transient <fields>;
  !private <fields>;
  !private <methods>;
  private void writeObject(java.io.ObjectOutputStream);
  private void readObject(java.io.ObjectInputStream);
  java.lang.Object writeReplace();
  java.lang.Object readResolve();
}

# 對於帶有回調函數的 onXXEvent、**On*Listener 的,不能被混淆
-keepclassmembers class * {
  void *(**On*Event);
  void *(**On*Listener);
}

# webView 處理,項目中沒有使用到 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, java.lang.String);
}

# js
-keepattributes JavascriptInterface
-keep class android.webkit.JavascriptInterface { *; }
-keepclassmembers class * {
  @android.webkit.JavascriptInterface <methods>;
}

# @Keep
-keep,allowobfuscation @interface android.support.annotation.Keep
-keep @android.support.annotation.Keep class *
-keepclassmembers class * {
  @android.support.annotation.Keep *;
}

一些自定義的模板

# 通配符*,匹配任意長度字符,但不含包名分隔符(.)
# 通配符**,匹配任意長度字符,並且包含包名分隔符(.)

# 不混淆某個類
-keep public class com.jasonwu.demo.Test { *; }

# 不混淆某個包所有的類
-keep class com.jasonwu.demo.test.** { *; }

# 不混淆某個類的子類
-keep public class * com.jasonwu.demo.Test { *; }

# 不混淆所有類名中包含瞭 ``model`` 的類及其成員
-keep public class **.*model*.** {*;}

# 不混淆某個接口的實現
-keep class * implements com.jasonwu.demo.TestInterface { *; }

# 不混淆某個類的構造方法
-keepclassmembers class com.jasonwu.demo.Test { 
 public <init>(); 
}

# 不混淆某個類的特定的方法
-keepclassmembers class com.jasonwu.demo.Test { 
 public void test(java.lang.String); 
}

aar中增加獨立的混淆配置

android {
  ···
  defaultConfig {
    ···
    consumerProguardFile 'proguard-rules.pro'
  }
  ···
}

檢查混淆和追蹤異常

開啟 Proguard 功能,則每次構建時 ProGuard 都會輸出下列文件:

dump.txt 說明 APK 中所有類文件的內部結構。

mapping.txt 提供原始與混淆過的類、方法和字段名稱之間的轉換。

seeds.txt 列出未進行混淆的類和成員。

usage.txt 列出從 APK 移除的代碼。

這些文件保存在 /build/outputs/mapping/release/ 中。我們可以查看 seeds.txt 裡面是否是我們需要保留的,以及 usage.txt 裡查看是否有誤刪除的代碼。 mapping.txt 文件很重要,由於我們的部分代碼是經過重命名的,如果該部分出現 bug,對應的異常堆棧信息裡的類或成員也是經過重命名的,難以定位問題。我們可以用 retrace 腳本(在 Windows 上為 retrace.bat;在 Mac/Linux 上為 retrace.sh)。它位於 /tools/proguard/ 目錄中。該腳本利用 mapping.txt 文件和你的異常堆棧文件生成沒有經過混淆的異常堆棧文件,這樣就可以看清是哪裡出問題瞭。使用 retrace 工具的語法如下:

retrace.bat|retrace.sh [-verbose] mapping.txt [<stacktrace_file>]

檢查混淆和追蹤異常

開啟 Proguard 功能,則每次構建時 ProGuard 都會輸出下列文件:

dump.txt 說明 APK 中所有類文件的內部結構。

mapping.txt 提供原始與混淆過的類、方法和字段名稱之間的轉換。

seeds.txt 列出未進行混淆的類和成員。

usage.txt 列出從 APK 移除的代碼。

這些文件保存在 /build/outputs/mapping/release/ 中。我們可以查看 seeds.txt 裡面是否是我們需要保留的,以及 usage.txt 裡查看是否有誤刪除的代碼。 mapping.txt 文件很重要,由於我們的部分代碼是經過重命名的,如果該部分出現 bug,對應的異常堆棧信息裡的類或成員也是經過重命名的,難以定位問題。我們可以用 retrace 腳本(在 Windows 上為 retrace.bat;在 Mac/Linux 上為 retrace.sh)。它位於 /tools/proguard/ 目錄中。該腳本利用 mapping.txt 文件和你的異常堆棧文件生成沒有經過混淆的異常堆棧文件,這樣就可以看清是哪裡出問題瞭。使用 retrace 工具的語法如下:

retrace.bat|retrace.sh [-verbose] mapping.txt [<stacktrace_file>]

結語

以上就是簡單介紹瞭Android中常用混淆的問題,還有很多不足的東西歡迎大傢補充 祝大傢新年快樂,技術越來越好,fighting

以上就是Android 一些常用的混淆Proguard的詳細內容,更多關於Android 常用的混淆Proguard的資料請關註WalkonNet其它相關文章!

推薦閱讀: