效能優化 (十一) ProGuard 對程式碼和資源壓縮

DevYK發表於2019-06-16

效能優化系列

APP 啟動優化

UI 繪製優化

記憶體優化

圖片壓縮

長圖優化

電量優化

Dex 加解密

動態替換 Application

APP 穩定性之熱修復原理探索

APP 持續執行之程式保活實現

ProGuard 對程式碼和資源壓縮

APK 極限壓縮

ProGuard 是什麼?

可以把 ProGuard 理解為是對程式碼和資源壓縮的一個工具,它能夠提供對 Java 類檔案的壓縮、優化、混淆,和預校驗。壓縮的步驟是檢測並移除未使用的類、欄位、方法和屬性。優化的步驟是分析和優化方法的位元組碼。混淆的步驟是使用短的毫無意義的名稱重新命名剩餘的類、欄位和方法。壓縮、優化、混淆使得程式碼更小,更高效。

AndroidStudio 怎麼使用 ProGuard ?

程式碼壓縮

要通過 ProGuard 啟用程式碼壓縮,請在 build.gradle 檔案內相應的構建型別中新增 minifyEnabled true

android {
    ...
    buildTypes {
        release {
            minifyEnabled true
            proguardFiles getDefaultProguardFile('proguard-android.txt'),
                   'proguard-rules.pro'
        }
    }
複製程式碼

ProGuard 會移除所有 (並且只會移除) 未使用的程式碼。不過 , ProGuard 難以對許多情況進行正確分析,可能會移除應用真正需要的程式碼。比如需要反射、動態載入所引用的類等情況,可能因為ProGuard 移除或者混淆了這部分沒使用的類,而導致錯誤。所以有時需要編寫混淆優化配置檔案。在 gradle 中的 proguardFiles 能夠讓我們傳遞File檔案或者檔案路徑交給 proguard 來執行。

配置 ProGuard 規則

現在我們開啟了混淆,但是還沒有配置混淆,我們可以在 build/intermediates/proguard-files/proguard-defaults.txt 來檢視預設的配置,現在我們可以根據預設的配置來進行我們專案的配置。

  • 分析預設配置檔案 proguard-defaults.txt-3.4.0

    # This is a configuration file for ProGuard.
    # http://proguard.sourceforge.net/index.html#manual/usage.html
    #
    # Starting with version 2.2 of the Android plugin for Gradle, this file is distributed together with
    # the plugin and unpacked at build-time. The files in $ANDROID_HOME are no longer maintained and
    # will be ignored by new version of the Android plugin for Gradle.
    
    # Optimizations can be turned on and off in the 'postProcessing' DSL block.
    # The configuration below is applied if optimizations are enabled.
    # Adding optimization introduces certain risks, since for example not all optimizations performed by
    # ProGuard works on all versions of Dalvik.  The following flags turn off various optimizations
    # known to have issues, but the list may not be complete or up to date. (The "arithmetic"
    # optimization can be used if you are only targeting Android 2.0 or later.)  Make sure you test
    # thoroughly if you go this route.
    
    ####################### START #######################
    
    # 混淆時所採用的演算法(谷歌推薦演算法)
    -optimizations !code/simplification/arithmetic,!code/simplification/cast,!field/*,!class/merging/*
    
    # 指定程式碼的壓縮級別(在0~7之間,預設為5)
    -optimizationpasses 5
    
    # 提高優化步驟
    -allowaccessmodification
    
    # 包名不混合大小寫
    -dontusemixedcaseclassnames
    
    # 不忽略非公共的庫類
    -dontskipnonpubliclibraryclasses
    
    # 輸出混淆日誌
    -verbose
    
    # 保持 Google 原生服務需要的類不被混淆
    -keep public class com.google.vending.licensing.ILicensingService
    -keep public class com.android.vending.licensing.ILicensingService
    -keep public class com.google.android.vending.licensing.ILicensingService
    -dontnote com.android.vending.licensing.ILicensingService
    -dontnote com.google.vending.licensing.ILicensingService
    -dontnote com.google.android.vending.licensing.ILicensingService
    
    # For native methods, see http://proguard.sourceforge.net/manual/examples.html#native
    # 混淆注意事項第二條,保持 native 方法不被混淆
    -keepclasseswithmembernames class * {
        native <methods>;
    }
    
    # Keep setters in Views so that animations can still work.
    # 保留自定義控制元件(繼承自View)不被混淆
    -keepclassmembers public class * extends android.view.View {
        void set*(***);
        *** get*();
    }
    
    # We want to keep methods in Activity that could be used in the XML attribute onClick.
    # 保留在 Activity 中的方法引數是 view 的方法(避免佈局檔案裡面 onClick 被影響)
    -keepclassmembers class * extends android.app.Activity {
        public void *(android.view.View);
    }
    
    # For enumeration classes, see http://proguard.sourceforge.net/manual/examples.html#enumerations
    # 保持列舉 enum 類不被混淆
    -keepclassmembers enum * {
        public static **[] values();
        public static ** valueOf(java.lang.String);
    }
    
    # 保持 Parcelable 序列化的類不被混淆(注:aidl 檔案不能去混淆)
    -keepclassmembers class * implements android.os.Parcelable {
        public static final ** CREATOR;
    }
    
    # 保持R(資源)下的所有類及其方法不能被混淆
    -keepclassmembers class **.R$* {
        public static <fields>;
    }
    
    # Preserve annotated Javascript interface methods.
    -keepclassmembers class * {
        @android.webkit.JavascriptInterface <methods>;
    }
    
    # 支援庫包含對較新版本版本的引用。
    # 不要警告那些情況下,這個應用程式連結到舊的
    # 平臺版本。我們知道他們是安全的。
    -dontnote android.support.**
    -dontnote androidx.**
    -dontwarn android.support.**
    -dontwarn androidx.**
    
    # 此類已棄用,但仍保留向後相容性。
    -dontwarn android.util.FloatMath
    
    # Support包規則
    # Understand the @Keep support annotation.
    -keep class android.support.annotation.Keep
    -keep class androidx.annotation.Keep
    
    -keep @android.support.annotation.Keep class * {*;}
    -keep @androidx.annotation.Keep class * {*;}
    
    # 保持 support Keep 類成員不被混淆
    -keepclasseswithmembers class * {
        @android.support.annotation.Keep <methods>;
    }
    
    # 保持 androidx Keep 類成員不被混淆
    -keepclasseswithmembers class * {
        @androidx.annotation.Keep <methods>;
    }
    
    # 保持 support Keep 類成員不被混淆
    -keepclasseswithmembers class * {
        @android.support.annotation.Keep <fields>;
    }
    
    # 保持 androidx Keep 類成員不被混淆
    -keepclasseswithmembers class * {
        @androidx.annotation.Keep <fields>;
    }
    
    # 不混淆所有類及其類成員中的使用註解的初始化方法
    -keepclasseswithmembers class * {
        @android.support.annotation.Keep <init>(...);
    }
    
    # 不混淆所有類及其類成員中的使用註解的初始化方法
    -keepclasseswithmembers class * {
        @androidx.annotation.Keep <init>(...);
    }
    
    # 排除 android.jar 和 org.apache.http.legacy.jar 之間重複
    -dontnote org.apache.http.**
    -dontnote android.net.http.**
    
    # 排除 android.jar 和核心-lambda-stubs.jar 之間重複。
    -dontnote java.lang.invoke.**
    
    複製程式碼

    以上就是預設配置檔案那麼這樣生成出來的 APK 到底是什麼樣的勒?下面放上一張圖片來看下。

    效能優化 (十一) ProGuard 對程式碼和資源壓縮

    發現上面的 類 已經變成不易閱讀的類了。下面我們就來單獨分析下 Proguard 的配置

基本指令
  • 指定程式碼的壓縮級別(在0~7之間,預設為5)

    -optimizationpasses 5
    複製程式碼
  • 是否使用大小寫混合(windows大小寫不敏感,建議加入)

    -dontusemixedcaseclassnames
    複製程式碼
  • 是否混淆非公共的庫的類

    -dontskipnonpubliclibraryclasses
    複製程式碼
  • 是否混淆非公共的庫的類的成員

    -dontskipnonpubliclibraryclassmembers
    複製程式碼
  • 混淆時是否做預校驗(Android不需要預校驗,去掉可以加快混淆速度)

    -dontpreverify
    複製程式碼
  • 混淆時是否記錄日誌(混淆後會生成對映檔案)

    -verbose
    複製程式碼
  • 指定外部模糊字典

    -obfuscationdictionary dictionary_path
    複製程式碼
  • 指定 class 模糊字典

    -classobfuscationdictionary dictionary_path
    複製程式碼
  • 指定 package 模糊字典

    -packageobfuscationdictionary dictionary_path
    複製程式碼
  • 混淆時所採用的演算法(谷歌推薦演算法)

    -optimizations !code/simplification/arithmetic,!field/,!class/merging/,!code/allocation/variable
    複製程式碼
  • 新增支援的jar(引入libs下的所有jar包)

    -libraryjars libs(*.jar;)
    複製程式碼
  • 將檔案來源重新命名為 SourceFile”`字串

    -renamesourcefileattribute SourceFile
    複製程式碼
  • 保持註解不被混淆

    -keepattributes Annotation
    -keep class * extends java.lang.annotation.Annotation {*;}
    -keep interface * extends java.lang.annotation.Annotation { *; }
    複製程式碼
  • 保持泛型不被混淆

    -keepattributes Signature-keep class * extends java.lang.annotation.Annotation {*;}
    複製程式碼
  • 保持反射不被混淆

    -keepattributes EnclosingMethod
    複製程式碼
  • 保持異常不被混淆

    -keepattributes Exceptions
    複製程式碼
  • 保持內部類不被混淆

    -keepattributes InnerClasses
    複製程式碼
  • 丟擲異常時保留程式碼行號

    -keepattributes SourceFile,LineNumberTable
    複製程式碼
keep 命令說明
指令 說明
-keep 保持類和類成員,防止被移除或者被重新命名
-keepnames 保持類和類成員,防止被重新命名
-keepclassmembers 保持類成員,防止被移除或者被重新命名
-keepclassmembernames 保持類成員,防止被重新命名
-keepclasseswithmembers 保持擁有該成員的類和成員,防止被移除或者被重新命名
-keepclasseswithmembernames 保持擁有該成員的類和成員,防止被重新命名
  • 保持元素不參與混淆的規則的命令格式:

    [保持命令] [類] {
        [成員]
    }
    
    具體的類
    訪問修飾符(publicprotectedprivate)
    萬用字元*,匹配任意長度字元,但不含包名分隔符(.)
    萬用字元**,匹配任意長度字元,並且包含包名分隔符(.)
    extends,即可以指定類的基類
    implement,匹配實現了某介面的類
    $,內部類
    “成員”代表類成員相關的限定條件,它將最終定位到某些符合該限定條件的類成員。它的內容可以使用:
    <init> 匹配所有構造器
    <fields> 匹配所有域
    <methods> 匹配所有方法
    萬用字元*,匹配任意長度字元,但不含包名分隔符(.)
    萬用字元**,匹配任意長度字元,並且包含包名分隔符(.)
    萬用字元***,匹配任意引數型別
    …,匹配任意長度的任意型別引數。比如void test(…)就能匹配任意 void test(String a) 或者是 void test(int a, String b) 這些方法。
    訪問修飾符(publicprotectedprivate複製程式碼
  • 不混淆某個類

    -keep public class com.example.proxy_core.ProxyApplication { *; }
    複製程式碼
  • 不混淆某個包所有的類

    -keep public class com.example.proxy_core.** { *; }
    複製程式碼
  • 不混淆某個類的子類

    -keep public class * extends class com.example.proxy_core.ProxyApplication { *; }
    複製程式碼
  • 不混淆所有類名中包含了 model的類及其成員

    -keep public class **.*model*.** {*;}
    複製程式碼
  • 不混淆某個介面的實現

    -keep class * implements com.example.proxy_core.LoadCallBack { *; }
    複製程式碼
  • 不混淆某個類的特定的方法

    -keepclassmembers class com.example.proxy_core.ProxyApplication {
         public void getVersion(java.lang.String);
    }
    複製程式碼
  • 不混淆某個類的內部類

    -keep class com.example.proxy_core.ProxyApplication$* {
           *;
    }
    複製程式碼
Proguard 注意事項
  • 基本元件不被混淆

    -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
    
    複製程式碼
  • 保持 Google 原生服務需要的類不被混淆

    -keep public class com.google.vending.licensing.ILicensingService
    -keep public class com.android.vending.licensing.ILicensingService
    
    複製程式碼
  • Support 包規則

    -dontwarn android.support.**
    -keep public class * extends android.support.v4.**
    -keep public class * extends android.support.v7.**
    -keep public class * extends android.support.annotation.**
    
    複製程式碼
  • 保持 native 方法不被混淆

    -keepclasseswithmembernames class * {
        native <methods>;
    }
    複製程式碼
  • 保留自定義控制元件 ( 繼承自 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);
        }
    複製程式碼
  • 保留指定格式的構造方法不被混淆

    -keepclasseswithmembers class * {
        public <init>(android.content.Context, android.util.AttributeSet);
        public <init>(android.content.Context, android.util.AttributeSet, int);
        }
    
    複製程式碼
  • 保留在 Activity 中的方法引數是 view 的方法(避免佈局檔案裡面 onClick 被影響)

    -keepclassmembers class * extends android.app.Activity {
        public void *(android.view.View);
    }
    複製程式碼
  • 保持列舉 enum 類不被混淆

    -keepclassmembers enum * {
        public static **[] values();
        public static ** valueOf(java.lang.String);
    }
    複製程式碼
  • 保持 R (資源)下的所有類及其方法不能被混淆

    -keep class **.R$* { *; }
    複製程式碼
  • 保持 Parcelable 序列化的類不被混淆(注:aidl 檔案不能去混淆)

    -keep class * implements android.os.Parcelable {
         public static final android.os.Parcelable$Creator *;
    }
    複製程式碼
  • 需要序列化和反序列化的類不能被混淆(注:Java 反射用到的類也不能被混淆)

    -keepnames class * implements java.io.Serializable
    複製程式碼
  • 保持 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();
    }
    
    複製程式碼
  • 保持 Adapter 類不被混淆

    -keep public class * extends android.widget.BaseAdapter { *; }
    複製程式碼
  • 保持 CusorAdapter 類不被混淆

    -keep public class * extends android.widget.CusorAdapter{ *; }
    複製程式碼
編寫基礎通用版本混淆規則
# This is a configuration file for ProGuard.
# http://proguard.sourceforge.net/index.html#manual/usage.html
#
# Starting with version 2.2 of the Android plugin for Gradle, this file is distributed together with
# the plugin and unpacked at build-time. The files in $ANDROID_HOME are no longer maintained and
# will be ignored by new version of the Android plugin for Gradle.

# Optimizations can be turned on and off in the 'postProcessing' DSL block.
# The configuration below is applied if optimizations are enabled.
# Adding optimization introduces certain risks, since for example not all optimizations performed by
# ProGuard works on all versions of Dalvik.  The following flags turn off various optimizations
# known to have issues, but the list may not be complete or up to date. (The "arithmetic"
# optimization can be used if you are only targeting Android 2.0 or later.)  Make sure you test
# thoroughly if you go this route.
# --------------------------------------------基本指令區-------------------------------------------# 指定程式碼的壓縮級別(在0~7之間,預設為5)
-optimizationpasses 5
# 是否使用大小寫混合(windows大小寫不敏感,建議加入)
-dontusemixedcaseclassnames
 # 是否混淆非公共的庫的類
-dontskipnonpubliclibraryclasses
# 是否混淆非公共的庫的類的成員
-dontskipnonpubliclibraryclassmembers
# 混淆時是否做預校驗(Android不需要預校驗,去掉可以加快混淆速度)
-dontpreverify
# 混淆時是否記錄日誌(混淆後會生成對映檔案)
-verbose

###################################### 如果有就新增 ############################################
#指定外部模糊字典
-obfuscationdictionary dictionary1.txt
#指定class模糊字典
-classobfuscationdictionary dictionary1.txt
#指定package模糊字典
-packageobfuscationdictionary dictionary2.txt
#########################################################################################

# 混淆時所採用的演算法(谷歌推薦演算法)
-optimizations !code/simplification/arithmetic,!field/*,!class/merging/*,!code/allocation/variable

# 新增支援的jar(引入libs下的所有jar包)
-libraryjars libs(*.jar;)

# 將檔案來源重新命名為“SourceFile”字串
-renamesourcefileattribute SourceFile

# 保持註解不被混淆
-keepattributes *Annotation*
-keep class * extends java.lang.annotation.Annotation {*;}

# 保持泛型不被混淆
-keepattributes Signature
# 保持反射不被混淆
-keepattributes EnclosingMethod
# 保持異常不被混淆
-keepattributes Exceptions
# 保持內部類不被混淆
-keepattributes Exceptions,InnerClasses
# 丟擲異常時保留程式碼行號
-keepattributes SourceFile,LineNumberTable

# --------------------------------------------預設保留區--------------------------------------------#
# 保持基本元件不被混淆
-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

# 保持 Google 原生服務需要的類不被混淆
-keep public class com.google.vending.licensing.ILicensingService
-keep public class com.android.vending.licensing.ILicensingService

# Support包規則
-dontwarn android.support.**
-keep public class * extends android.support.v4.**
-keep public class * extends android.support.v7.**
-keep public class * extends android.support.annotation.**

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

# 保留自定義控制元件(繼承自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);
}

# 保留指定格式的構造方法不被混淆
-keepclasseswithmembers class * {
    public <init>(android.content.Context, android.util.AttributeSet);
    public <init>(android.content.Context, android.util.AttributeSet, int);
}

# 保留在Activity中的方法引數是view的方法(避免佈局檔案裡面onClick被影響)
-keepclassmembers class * extends android.app.Activity {
    public void *(android.view.View);
}

# 保持列舉 enum 類不被混淆
-keepclassmembers enum * {
    public static **[] values();
    public static ** valueOf(java.lang.String);
}

# 保持R(資源)下的所有類及其方法不能被混淆
-keep class **.R$* { *; }

# 保持 Parcelable 序列化的類不被混淆(注:aidl檔案不能去混淆)
-keep class * implements android.os.Parcelable {
    public static final android.os.Parcelable$Creator *;
}

# 需要序列化和反序列化的類不能被混淆(注:Java反射用到的類也不能被混淆)
-keepnames class * implements java.io.Serializable

# 保持 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();
}

# 保持 BaseAdapter 類不被混淆
-keep public class * extends android.widget.BaseAdapter { *; }
# 保持 CusorAdapter 類不被混淆
-keep public class * extends android.widget.CusorAdapter{ *; }

# --------------------------------------------webView區--------------------------------------------#
# WebView處理,專案中沒有使用到webView忽略即可
# 保持Android與JavaScript進行互動的類不被混淆
-keep class **.AndroidJavaScript { *; }
-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.WebChromeClient {
     public void *(android.webkit.WebView,java.lang.String);
}

# 網路請求相關
-keep public class android.net.http.SslError

# --------------------------------------------刪除程式碼區--------------------------------------------#
# 刪除程式碼中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(...);
}
複製程式碼
第三方需要的混淆規則
################alipay###############

-keep class com.alipay.android.app.IAlixPay{*;}
-keep class com.alipay.android.app.IAlixPay$Stub{*;}
-keep class com.alipay.android.app.IRemoteServiceCallback{*;}
-keep class com.alipay.android.app.IRemoteServiceCallback$Stub{*;}
-keep class com.alipay.sdk.app.PayTask{ public *;}
-keep class com.alipay.sdk.app.AuthTask{ public *;}

################retrofit###############
-dontwarn retrofit2.**
-keep class retrofit2.** { *; }
-keepattributes Signature
-keepattributes Exceptions

################butterknife###############
-keep class butterknife.** { *; }
-dontwarn butterknife.internal.**
-keep class **$$ViewBinder { *; }
-keepclasseswithmembernames class * {
   @butterknife.* <fields>;
}
-keepclasseswithmembernames class * {
 @butterknife.* <methods>;
}


################gson###############
-keepattributes Signature
-keepattributes *Annotation*
-keep class sun.misc.Unsafe { *; }
-keep class com.google.gson.stream.** { *; }
# Application classes that will be serialized/deserialized over Gson
-keep class com.sunloto.shandong.bean.** { *; }


################glide###############
-keep public class * implements com.bumptech.glide.module.AppGlideModule
-keep public class * implements com.bumptech.glide.module.LibraryGlideModule
-keep class com.bumptech.glide.** { *; }
-keep public enum com.bumptech.glide.load.resource.bitmap.ImageHeaderParser$** {
    **[] $VALUES;
    public *;
}

################okhttp###############
-keepattributes Signature
-keepattributes *Annotation*
-keep class com.squareup.okhttp.** { *; }
-keep interface com.squareup.okhttp.** { *; }
-keep class okhttp3.** { *; }
-keep interface okhttp3.** { *; }
-dontwarn com.squareup.okhttp.**


################androidEventBus###############
-keep class org.simple.** { *; }
-keep interface org.simple.** { *; }
-keepclassmembers class * {
    @org.simple.eventbus.Subscriber <methods>;
}
-keepattributes *Annotation*


################EventBus###############
-keepclassmembers class * {
    @org.greenrobot.eventbus.Subscribe <methods>;
}
-keep class org.greenrobot.eventbus.EventBus { *; }
-keep enum org.greenrobot.eventbus.ThreadMode { *; }

-keepclassmembers class * extends org.greenrobot.eventbus.util.ThrowableFailureEvent {
    <init>(java.lang.Throwable);
}

################autolayout###############
-keep class com.zhy.autolayout.** { *; }
-keep interface com.zhy.autolayout.** { *; }


################RxJava and RxAndroid###############
-dontwarn org.mockito.**
-dontwarn org.junit.**
-dontwarn org.robolectric.**

-keep class io.reactivex.** { *; }
-keep interface io.reactivex.** { *; }

-keepattributes Signature
-keepattributes *Annotation*
-keep class com.squareup.okhttp.** { *; }
-dontwarn okio.**
-keep interface com.squareup.okhttp.** { *; }
-dontwarn com.squareup.okhttp.**

-dontwarn io.reactivex.**
-dontwarn retrofit.**
-keep class retrofit.** { *; }
-keepclasseswithmembers class * {
    @retrofit.http.* <methods>;
}

-keep class sun.misc.Unsafe { *; }

-dontwarn java.lang.invoke.*

-keep class io.reactivex.schedulers.Schedulers {
    public static <methods>;
}
-keep class io.reactivex.schedulers.ImmediateScheduler {
    public <methods>;
}
-keep class io.reactivex.schedulers.TestScheduler {
    public <methods>;
}
-keep class io.reactivex.schedulers.Schedulers {
    public static ** test();
}
-keepclassmembers class io.reactivex.internal.util.unsafe.*ArrayQueue*Field* {
    long producerIndex;
    long consumerIndex;
}
-keepclassmembers class io.reactivex.internal.util.unsafe.BaseLinkedQueueProducerNodeRef {
    long producerNode;
    long consumerNode;
}

-keepclassmembers class io.reactivex.internal.util.unsafe.BaseLinkedQueueProducerNodeRef {
    io.reactivex.internal.util.atomic.LinkedQueueNode producerNode;
}
-keepclassmembers class io.reactivex.internal.util.unsafe.BaseLinkedQueueConsumerNodeRef {
    io.reactivex.internal.util.atomic.LinkedQueueNode consumerNode;
}

-dontwarn io.reactivex.internal.util.unsafe.**



################espresso###############
-keep class android.support.test.espresso.** { *; }
-keep interface android.support.test.espresso.** { *; }



################annotation###############
-keep class android.support.annotation.** { *; }
-keep interface android.support.annotation.** { *; }


################RxLifeCycle#################
-keep class com.trello.rxlifecycle2.** { *; }
-keep interface com.trello.rxlifecycle2.** { *; }


################RxPermissions#################
-keep class com.tbruyelle.rxpermissions2.** { *; }
-keep interface com.tbruyelle.rxpermissions2.** { *; }

################RxCache#################
-dontwarn io.rx_cache2.internal.**
-keep class io.rx_cache2.internal.Record { *; }
-keep class io.rx_cache2.Source { *; }

-keep class io.victoralbertos.jolyglot.** { *; }
-keep interface io.victoralbertos.jolyglot.** { *; }

################RxErrorHandler#################
 -keep class me.jessyan.rxerrorhandler.** { *; }
 -keep interface me.jessyan.rxerrorhandler.** { *; }

################Timber#################
-dontwarn org.jetbrains.annotations.**


################Canary#################
-dontwarn com.squareup.haha.guava.**
-dontwarn com.squareup.haha.perflib.**
-dontwarn com.squareup.haha.trove.**
-dontwarn com.squareup.leakcanary.**
-keep class com.squareup.haha.** { *; }
-keep class com.squareup.leakcanary.** { *; }

# Marshmallow removed Notification.setLatestEventInfo()
-dontwarn android.app.Notification

複製程式碼
混淆後的程式碼錯誤棧恢復方法
  1. 把 outputs/mapping/debug/mapping.txt 檔案儲存 (儲存了混淆前後的對應關係)。
  2. 使用工具 sdk/tools/groguard/bin/retrace.bat 先配置 -keepattributes SourceFile,LineNumberTable 再執行 retrace.bat -verbose mappint 檔案 bug 檔案

資源壓縮

資源壓縮只與程式碼壓縮協同工作。程式碼壓縮器移除所有未使用的程式碼後,資源壓縮器便可確定應用仍然使用的資源。這在您新增包含資源的程式碼庫時體現得尤為明顯 - 您必須移除未使用的庫程式碼,使庫資源變為未引用資源,才能通過資源壓縮器將它們移除

要啟用資源壓縮,請在 build.gradle 檔案中將 shrinkResources 屬性設定為 true(在用於程式碼壓縮的 minifyEnabled 旁邊)。例如:

android {
    ...
    buildTypes {
        release {
            shrinkResources true
            minifyEnabled true
            proguardFiles getDefaultProguardFile('proguard-android.txt'),
                    'proguard-rules.pro'
        }
    }
}
複製程式碼

如果您尚未使用程式碼壓縮用途的 minifyEnabled 構建應用,請先嚐試使用它,然後再啟用 shrinkResources,因為您可能需要編輯 proguard-rules.pro 檔案以保留動態建立或呼叫的類或方法,然後再開始移除資源。

移除未使用的備用資源
  • 移除僅支援中文和英語

    android {
        defaultConfig {
            ...
            resConfigs "zh","en"
        }
    }
    複製程式碼
  • 合併重複資源

    預設情況下,Gradle 還會合並同名資源,例如可能位於不同資原始檔夾中的同名可繪製物件。這一行為不受 shrinkResources 屬性控制,也無法停用,因為在有多個資源匹配程式碼查詢的名稱時,有必要利用這一行為來避免錯誤。

    只有在兩個或更多個檔案具有完全相同的資源名稱、型別和限定符時,才會進行資源合併。Gradle 會在重複項中選擇其視為最佳選擇的檔案(根據下述優先順序),並只將這一個資源傳遞給 AAPT,以供在 APK 檔案中分發。

    Gradle 會在下列位置尋找重複資源:

    • 與主源集關聯的主資源,一般位於 src/main/res/ 中。
    • 變體疊加,來自構建型別和構建風味。
    • 庫專案依賴項。

    Gradle 會按以下級聯優先順序合併重複資源:

    依賴項 → 主資源 → 構建風味 → 構建型別

    例如,如果某個重複資源同時出現在主資源和構建風味中,Gradle 會選擇構建風味中的重複資源。

    如果完全相同的資源出現在同一源集中,Gradle 無法合併它們,並且會發出資源合併錯誤。如果您在 build.gradle 檔案的 sourceSet 屬性中定義了多個源集,則可能會發生這種情況,例如,如果 src/main/res/src/main/res2/ 包含完全相同的資源,就可能會發生這種情況。

  • 排查資源壓縮問題

    當您壓縮資源時,Gradle Console 會顯示它從應用軟體包中移除的資源的摘要。例如:

    :android:shrinkDebugResources Removed unused resources: Binary resource data reduced from 2570KB to 1711KB: Removed 33% :android:validateDebugSigning
    複製程式碼

    Gradle 還會在 <module-name>/build/outputs/mapping/release/(ProGuard 輸出檔案所在的資料夾)中建立一個名為 resources.txt 的診斷檔案。該檔案包括諸如哪些資源引用了其他資源以及使用或移除了哪些資源等詳情。

感謝:

壓縮程式碼和資源

相關文章