如何在10分鐘內(nèi)讓應(yīng)用大小減少60%?
一個(gè)APP的包之所以大,主要包括一下文件
- 代碼
- lib
- so本地庫
- 資源文件(圖片,音頻,字體等)
瘦身就主要瘦這些。
一、打包的時(shí)候刪除不用的代碼
- buildTypes {
- debug {
- ...
- shrinkResources true // 是否去除無效的資源文件(如果你的Debug也需要瘦身)
- }
- release {
- ...
- shrinkResources true // 是否去除無效的資源文件
- }
- }
二、減少不必要的打包
- defaultConfig {
- ...
- //打包的語言類型(語種的翻譯)
- resConfigs "en", "de", "fr", "it"
- //打包的文件夾
- resConfigs "nodpi", "hdpi", "xhdpi", "xxhdpi", "xxxhdpi"
- }
或者
- android {
- ...
- splits {
- density {
- enable true
- exclude "ldpi", "tvdpi", "xxxhdpi"
- compatibleScreens 'small', 'normal', 'large', 'xlarge'
- //reset()
- //include 'x86', 'armeabi-v7a', 'mips'
- //universalApk true
- }
- }
三、lib
盡量不用太復(fù)雜的lib,輕量級lib是首選。如果你的應(yīng)用沒用到兼容庫,可以考慮去掉support包。
四、資源文件
我們可以通過Lint工具找到?jīng)]有使用的資源(在Android Studio的“Analyze”菜單中選擇“Inspect Code…”)
五、把現(xiàn)有圖片轉(zhuǎn)換為webP
我們可以通過 智圖 或者isparta將其它格式的圖片轉(zhuǎn)換成webP格式,isparta可實(shí)現(xiàn)批量轉(zhuǎn)換。
六、圖片相關(guān)
在Android 5.0及以上的版本可以通過tintcolor實(shí)現(xiàn)只提供一張按鈕的圖片,在程序中實(shí)現(xiàn)按鈕反選效果,前提是圖片的內(nèi)容一樣,只是正反選按鈕的顏色不一樣。
- Drawable.setColorFilter( 0xffff0000, Mode.MULTIPLY )
在Android 5.0及以上的版本,可以使用VectorDrawable和SVG圖片來替換原有圖片
七、混淆
1 構(gòu)建多個(gè)版本
在gradle中的buildTypes中增加不同的構(gòu)建類型,使用applicationSuffix和versionNameSuffix可以生成多個(gè)版本在同一設(shè)備上運(yùn)行
創(chuàng)建src/[buildType]/res/設(shè)置不同的ic_launcher以區(qū)別不同版本
2 混淆參數(shù)
- {
- debug { minifyEnabled false }
- release {
- signingConfig signingConfigs.release
- minifyEnabled true
- proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
- }
- }
minifyEnabled true
- 是否要啟用通過 ProGuard 實(shí)現(xiàn)的代碼壓縮(true啟用)
- 請注意,代碼壓縮會(huì)拖慢構(gòu)建速度,因此您應(yīng)該盡可能避免在調(diào)試構(gòu)建中使用。
注:Android Studio 會(huì)在使用Instant Run時(shí)停用 ProGuard。
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
getDefaultProguardFile(‘proguard-android.txt')方法可從 Android SDKtools/proguard/文件夾獲取默認(rèn) ProGuard 設(shè)置。
提示:要想做進(jìn)一步的代碼壓縮,可嘗試使用位于同一位置的proguard-android-optimize.txt文件。它包括相同的 ProGuard 規(guī)則,但還包括其他在字節(jié)碼一級(方法內(nèi)和方法間)執(zhí)行分析的優(yōu)化,以進(jìn)一步減小 APK 大小和幫助提高其運(yùn)行速度。
proguard-rules.pro文件用于添加自定義 ProGuard 規(guī)則。默認(rèn)情況下,該文件位于模塊根目錄(build.gradle文件旁)。
要添加更多各構(gòu)建變體專用的 ProGuard 規(guī)則,請?jiān)谙鄳?yīng)的productFlavor代碼塊中再添加一個(gè)proguardFiles屬性。例如,以下 Gradle 文件會(huì)向flavor2產(chǎn)品風(fēng)味添加flavor2-rules.pro。現(xiàn)在flavor2使用所有三個(gè) ProGuard 規(guī)則,因?yàn)檫€應(yīng)用了來自release代碼塊的規(guī)則。
每次構(gòu)建時(shí) ProGuard 都會(huì)輸出下列文件 dump.txt 說明 APK 中所有類文件的內(nèi)部結(jié)構(gòu)。mapping.txt:提供原始與混淆過的類、方法和字段名稱之間的轉(zhuǎn)換。seeds.txt:列出未進(jìn)行混淆的類和成員。usage.txt:列出從 APK 移除的代碼。這些文件保存在/build/outputs/mapping/release/。
要修正錯(cuò)誤并強(qiáng)制 ProGuard 保留特定代碼,請?jiān)?ProGuard 配置文件中添加一行-keep代碼。例如:
- -keeppublicclassMyClass
您還可以向您想保留的代碼添加[@Keep]
(https://developer.android.com/reference/android/support/annotation/Keep.html)注解。在類上添加@Keep可原樣保留整個(gè)類。在方法或字段上添加它可完整保留方法/字段(及其名稱)以及類名稱。請注意,只有在使用注解支持庫時(shí),才能使用此注解。
在使用-keep選項(xiàng)時(shí),有許多事項(xiàng)需要考慮;如需了解有關(guān)自定義配置文件的詳細(xì)信息,請閱讀ProGuard 手冊。問題排查一章概述了您可能會(huì)在混淆代碼時(shí)遇到的其他常見問題。
請注意,您每次使用 ProGuard 創(chuàng)建發(fā)布構(gòu)建時(shí)都會(huì)覆蓋mapping.txt文件,因此您每次發(fā)布新版本時(shí)都必須小心地保存一個(gè)副本。通過為每個(gè)發(fā)布構(gòu)建保留一個(gè)mapping.txt文件副本,您就可以在用戶提交的已混淆堆疊追蹤來自舊版本應(yīng)用時(shí)對問題進(jìn)行調(diào)試。
在每次添加庫的時(shí)候,需要及時(shí)進(jìn)行make a release build
DexGuard時(shí)Proguard同一個(gè)團(tuán)隊(duì)開發(fā)的軟件, 優(yōu)化代碼,分離dex文件從而解決65k方法限制的文件
關(guān)于proguard-android.txt文件:
- -dontusemixedcaseclassnames: 表示混淆時(shí)不使用大小寫混淆類名。
- -dontskipnonpubliclibraryclasses:不跳過library中的非public方法。
- -verbose: 打印混淆的詳細(xì)信息。
- -dontoptimize: 不進(jìn)行優(yōu)化,優(yōu)化可能會(huì)造成一些潛在風(fēng)險(xiǎn),不能保證在所有版本的Dalvik上都正常運(yùn)行。
- -dontpreverify: 不進(jìn)行預(yù)校驗(yàn)。
- -keepattributes Annotation :對注解參數(shù)進(jìn)行保留。
- -keep public class com.google.vending.licensing.ILicensingService
- -keep public class com.android.vending.licensing.ILicensingService:
表示不混淆上述聲明的兩個(gè)類。
proguard中一共有三組六個(gè)keep關(guān)鍵字的含義
- keep 保留類和類中的成員,防止它們被混淆或移除。
- keepnames 保留類和類中的成員,防止它們被混淆,但當(dāng)成員沒有被引用時(shí)會(huì)被移除。
- keepclassmembers 只保留類中的成員,防止它們被混淆或移除。
- keepclassmembernames 只保留類中的成員,防止它們被混淆,但當(dāng)成員沒有被引用時(shí)會(huì)被移除。
- keepclasseswithmembers 保留類和類中的成員,防止它們被混淆或移除,前提是指名的類中的成員必須存在,如果不存在則還是會(huì)混淆。
- keepclasseswithmembernames 保留類和類中的成員,防止它們被混淆,但當(dāng)成員沒有被引用時(shí)會(huì)被移除,前提是指名的類中的成員必須存在,如果不存在則還是會(huì)混淆。
keepclasseswithmember和keep關(guān)鍵字的區(qū)別:
- 如果這個(gè)類沒有native的方法,那么這個(gè)類會(huì)被混淆
- -keepclasseswithmember class * {
- native <methods>;
- }
- 不管這個(gè)類有沒有native的方法,那么這個(gè)類不會(huì)被混淆
- -keep class * {
- native <methods>;
- }
另外、 你可以使用 APK Analyser 分解你的 APK
Android Studio 提供了一個(gè)有用的工具:APK Analyser。APK Analyser 將會(huì)拆解你的應(yīng)用并讓你知道 .apk 文件中的那個(gè)部分占據(jù)了大量空間。讓我們看一下 Anti-Theft 在沒有經(jīng)過優(yōu)化之前的截圖。
從 Apk Analyser 的輸出來看,應(yīng)用的原大小是 3.1MB。經(jīng)過 Play 商店的壓縮,大致是 2.5MB。
從截圖中可以看出主要有 3 個(gè)文件夾占據(jù)了應(yīng)用的大多數(shù)空間。
- classes.dex —— 這是 dex 文件,包含了所有會(huì)運(yùn)行在你的 DVM 或 ART 里的字節(jié)碼文件。
- res —— 這個(gè)文件夾包含了所有在 res 文件夾下的文件。大部分情況下它包含所有圖片,圖標(biāo)和源文件,菜單文件和布局。
- resources.arsc —— 這個(gè)文件包含了所有 value 資源。這個(gè)文件包含了你 value 目錄下的所有數(shù)據(jù)。包括 strings、dimensions、styles、intergers、ids 等等。
你有兩個(gè)默認(rèn)的混淆文件。
- proguard-android-optimize.txt
- proguard-android.txt
就像文件名寫的那樣,“proguard-android-optimize.txt”是更積極的混淆選項(xiàng)。我們將這個(gè)作為默認(rèn)的混淆配置。你可以在 /app 目錄下的 proguard-rules.pro 里添加自定義的混淆配置。
- release {
- //Enable the proguard
- minifyEnabled true
- proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), "proguard-rules.pro"
- //Other parameters
- debuggable false
- jniDebuggable false
- renderscriptDebuggable false
- signingConfig playStoreConfig //Add your own signing config
- pseudoLocalesEnabled false
- zipAlignEnabled true
通過設(shè)置 minifyEnabled 為 true,混淆將會(huì)移除所有未使用的方法、指令以減小 classes.dex 文件。
這是啟用了 minify 之后的 APK。
八、AndroidStudio使用lint清除無用的資源文件
在使用AndroidStudio進(jìn)行App開發(fā)的時(shí)候,我們經(jīng)常會(huì)在項(xiàng)目中引用多種資源文件,包括圖片,布局文件,常量引用定義。隨著項(xiàng)目版本開發(fā)的迭代,每一期的資源會(huì)有變動(dòng)必定會(huì)留下一些無用的資源這個(gè)時(shí)候我們手動(dòng)去一個(gè)一個(gè)尋找效率就會(huì)很低下。這個(gè)時(shí)候我們就要學(xué)會(huì)AndroidStudio使用lint清除無用的資源文件。
打開AndroidStudio在項(xiàng)目中,點(diǎn)擊最上方的菜單欄Analyze -> Run Inspection by Name 如下圖:
點(diǎn)擊 Run Inspection by Name會(huì)彈出一個(gè)對話框。在對話框里面輸入unused resource 如下圖:
然后點(diǎn)擊下拉列表中的unused resource。 之后會(huì)彈出一個(gè)對話框如下圖