類比於微信,如何對Apk進行極限壓縮,談下Android壓縮8大步

面試官: 類比於微信,如何對Apk進行極限壓縮,談下Android壓縮8大步

心理分析:面試官是非常注重性能優化的,考求職者是否具備APK壓縮技能。

**求職者:**應該從每一步壓縮開始,壓縮的過程本質是擠牙膏的過程,一步一步擠。將Apk壓縮分為8個步驟

簡介

隨著項目的不斷迭代,代碼量跟資源文件不斷增多。那麼就會出現打包後的 APK 文件越來越大,如果突然有一天你們老闆或領導叫你優化 APK 大小,你還不知道怎麼優化那就有點說不過去了,這篇文章咱們就來一起分析並優化 APK 體積大小吧。

分析 APK 資源佔用

注意:

我是在 GitHub 找了一個人氣比較高的開源項目,需要的話自己可以點擊下載,自己動手嘗試一番.

類比於微信,如何對Apk進行極限壓縮,談下Android壓縮8大步

分析工具直接用的 AS Build/Analyze APK

從上面圖中得出 assets > classes.dex > res > lib 其中資源文件佔用最大。

下面我們就來看看怎麼減小 APK 大小吧,

優化 APK 體積八大步

1. 將圖片轉換為 webp 格式

Webp 概念

WebP 是一種同時提供了有損壓縮與無損壓縮的圖片文件格式,派生自視頻編碼格式 VP8。WebP 最初在2010年發佈,目標是減少文件大小,但達到 和 JEPG 格式相同的圖片質量,希望能夠減少圖片檔在網絡上的發送時間。2011年11月8日,Google 開始讓 WebP 支持無損壓縮和透明色的功能。

根據 Google 較早的測試,WebP 的無損壓縮比網絡上找到的 PNG 檔少了 45% 的文件大小,即使這些 PNG 檔在使用 PNGCRUSH 和 PNGOUT 處理過,WebP 還是可以減少 28% 的文件大小。就目前而言,Webp 可以讓圖片大小平均減少 70% 。WebP 是未來圖片格式的發展趨勢。

PNG / JPG to Webp

類比於微信,如何對Apk進行極限壓縮,談下Android壓縮8大步

點擊圖片或者文件夾右鍵選擇 Convert to Webp 格式,將 png / jpg 圖片壓縮為 webp 格式圖片.

類比於微信,如何對Apk進行極限壓縮,談下Android壓縮8大步

最後我們只減少了不到 200 kb 左右,有可能項目圖片資源本來就沒有多大,只是太多小圖片導致的。

應用場景及優勢

  • 客戶端軟件,內嵌了基於 Chromium 的 webview,這類瀏覽器中應用的網頁是可以完全使用WebP 格式,提升加載渲染速度,不考慮兼容。
  • 用 node-webkit 開發的程序,用 WebP 可以減少文件包的體積。
  • 移動應用 或 網頁遊戲 ,界面需要大量圖片,可以嵌入 WebP 的解碼包,能夠節省用戶流量,提升訪問速度優勢:
  • 對於 PNG 圖片,WebP 比 PNG 小了45%。

2. 去除多語言

在 app/build.gradle 添加

android{
 ...
 defaultConfig{
 ...
 //只保留英語
 resConfigs "en"
 }
}
複製代碼
類比於微信,如何對Apk進行極限壓縮,談下Android壓縮8大步

這裡我們發現減少了大概 200 kb

3. 去除不必要 so 庫

通過反編譯 Android 微信版本 得知,微信也只適配了 armeabi-v7a 架構,那麼我們刪掉其它庫的支持吧。

android{
 ...
 defaultConfig{
 ...
 ndk {
 //設置支持的SO庫架構
 abiFilters "armeabi-v7a"
 }
}
}
複製代碼
類比於微信,如何對Apk進行極限壓縮,談下Android壓縮8大步

又優化了差不多 600 kb ,繼續。

4. 去除無用資源 Link 檢查(謹慎刪除)

概念

Lint 是 Android Studio 提供的 代碼掃描分析工具,它可以幫助我們發現代碼結構 / 質量問題,同時提供一些解決方案,而且這個過程不需要我們手寫測試用例。代碼迭代版本一多,很容易會遺留一些無用的代碼、資源文件,我們可以使用 Lint 進行清除。

怎麼使用 Link 檢查

打開 AS 工具,找到 Analyze > Run Inspection By Name > unused resources

優化

類比於微信,如何對Apk進行極限壓縮,談下Android壓縮8大步

發現我們 link 大概優化了 700 kb繼續。

注意

因為 link 是檢查有沒有引用來做的判斷是否使用了資源,那麼如果是這種方式勒,所以在刪除的時候一定要謹慎。

//動態獲取資源 id , 未直接使用 R.xx.xx ,則這個 id 代表的資源會被認為沒有使用過(類似不能混淆反射類)
int indetifier =getResources().getIdentifier("img_bubble_receive", "drawable", getPackageName()); getResources().getDrawable(indetifier);
複製代碼

5. 開啟混淆

類比於微信,如何對Apk進行極限壓縮,談下Android壓縮8大步

優化了大概 1.7M 繼續。

6.移除無用資源 shinkResource

  • 開啟 shinkResource = true
 buildTypes {
 release {
 minifyEnabled true
 shrinkResources = true
 proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
 }
 debug {
 shrinkResources = true
 minifyEnabled true
 proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
 }
 }
複製代碼
類比於微信,如何對Apk進行極限壓縮,談下Android壓縮8大步

這個有可能 link 刪除了無用資源,所以沒有在優化了

7.開啟刪除無用資源 (嚴格模式和普通模式) - 這個我這裡就不可測試,你們下來可以測試下效果

普通模式也就是自定義模式

如果您有想要保留或捨棄的特定資源,請在您的項目中創建一個包含 標記的 XML 文件,並在 tools:keep 屬性中指定每個要保留的資源,在 tools:discard 屬性中指定每個要捨棄的資源。這兩個屬性都接受逗號分隔的資源名稱列表。您可以使用星號字符作為通配符。

例如:


複製代碼

將該文件保存在項目資源中,例如,保存在 res/raw/keep.xml。構建不會將該文件打包到 APK 之中。

指定要捨棄的資源可能看似愚蠢,因為您本可將它們刪除,但在使用構建變體時,這樣做可能很有用。例如,如果您明知給定資源表面上會在代碼中使用(並因此不會被壓縮器移除),但實際不會用於給定構建變體,就可以將所有資源放入公用項目目錄,然後為每個構建變體創建一個不同的 keep.xml 文件。構建工具也可能無法根據需要正確識別資源,這是因為編譯器會添加內聯資源 ID,而資源分析器可能不知道真正引用的資源和恰巧具有相同值的代碼中的整數值之間的差別。

嚴格模式

正常情況下,資源壓縮器可準確判定系統是否使用了資源。不過,如果您的代碼調用 Resources.getIdentifier()(或您的任何庫進行了這一調用 - AppCompat 庫會執行該調用),這就表示您的代碼將根據動態生成的字符串查詢資源名稱。當您執行這一調用時,默認情況下資源壓縮器會採取防禦性行為,將所有具有匹配名稱格式的資源標記為可能已使用,無法移除。

例如,以下代碼會使所有帶 img_ 前綴的資源標記為已使用。

String name = String.format("img_%1d", angle + 1);
res = getResources().getIdentifier(name, "drawable", getPackageName());
複製代碼

資源壓縮器還會瀏覽代碼以及各種 res/raw/ 資源中的所有字符串常量,尋找格式類似於
file:///android_res/drawable//ic_plus_anim_016.png 的資源網址。如果它找到與其類似的字符串,或找到其他看似可用來構建與其類似的網址的字符串,則不會將它們移除。

這些是默認情況下啟用的安全壓縮模式的示例。但您可以停用這一“有備無患”處理方式,並指定資源壓縮器只保留其確定已使用的資源。要執行此操作,請在 keep.xml 文件中將 shrinkMode 設置為 strict,如下所示:


複製代碼

如果您確已啟用嚴格壓縮模式,並且代碼也引用了包含動態生成字符串的資源(如上所示),則必須利用 tools:keep 屬性手動保留這些資源。

8. AndResGuard 微信資源壓縮方案

什麼是 AndResGuard

AndResGuard 是一個縮小 APK 大小的工具,它的原理類似 Java Proguard ,但是隻針對資源。它會將原本冗長的資源路徑變短,例如將 res/drawable/wechat 變為 r/d/a。

為什麼使用 AndResGuard

在以往的開發中,我們通常只混淆了代碼,資源文件卻暴露在他人面前,res 文件夾下所有文件名的可讀性過強。

使用後的效果

類比於微信,如何對Apk進行極限壓縮,談下Android壓縮8大步

AndResGuard 的配置

  • 項目根目錄下 build.gradle 中,添加插件的依賴:
 dependencies {
 classpath 'com.tencent.mm:AndResGuard-gradle-plugin:1.2.16'
 }
複製代碼
  • 在 app 目錄下,創建 and_res_guard.gradle 文件
apply plugin: 'AndResGuard'
andResGuard {
 mappingFile = null
 use7zip = true
 useSign = true
 keepRoot = false
 compressFilePattern = [
 "*.png",
 "*.jpg",
 "*.jpeg",
 "*.gif",
 "resources.arsc"
 ]
 whiteList = [
 // your icon
 "R.drawable.icon",
 // for fabric
 "R.string.com.crashlytics.*",
 // for umeng update
 "R.string.tb_*",
 "R.layout.tb_*",
 "R.drawable.tb_*",
 "R.drawable.u1*",
 "R.drawable.u2*",
 "R.color.tb_*",
 // umeng share for sina
 "R.drawable.sina*",
 // for google-services.json
 "R.string.google_app_id",
 "R.string.gcm_defaultSenderId",
 "R.string.default_web_client_id",
 "R.string.ga_trackingId",
 "R.string.firebase_database_url",
 "R.string.google_api_key",
 "R.string.google_crash_reporting_api_key",
 //友盟
 "R.string.umeng*",
 "R.string.UM*",
 "R.layout.umeng*",
 "R.drawable.umeng*",
 "R.id.umeng*",
 "R.anim.umeng*",
 "R.color.umeng*",
 "R.style.*UM*",
 "R.style.umeng*",
 //融雲
 "R.drawable.u*",
 "R.drawable.rc_*",
 "R.string.rc_*",
 "R.layout.rc_*",
 "R.color.rc_*",
 "R.id.rc_*",
 "R.style.rc_*",
 "R.dimen.rc_*",
 "R.array.rc_*"
 ]
 sevenzip {
 artifact = 'com.tencent.mm:SevenZip:1.2.10'
 }
}
複製代碼
  • 在 app 模塊下的 build.gradle 文件添加
apply from: 'and_res_guard.gradle'
複製代碼
  • 打包完之後效果圖
類比於微信,如何對Apk進行極限壓縮,談下Android壓縮8大步

類比於微信,如何對Apk進行極限壓縮,談下Android壓縮8大步

資源壓縮了大概 1M

總結

  1. 項目體積越大,資源越多,效果就越明顯。
  2. 使用 Link 刪除資源的話,一定要謹慎,提前做好備份。
  3. 咱們這裡因為項目本身只有 22 M 多,最後優化了 4.5 M 下去。也還是很不容易的。


分享到:


相關文章: