2M-APK瘦身实测可行又便捷的方法 (应用上架 阵前磨枪)

2023-12-18 12:59

本文主要是介绍2M-APK瘦身实测可行又便捷的方法 (应用上架 阵前磨枪),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

新项目上线,新应用原始apk大小在6.9M左右,然后公司和第三方运营公司合作,加入其提供的第三方SDK,瞬间apk体积增大3M;接着为了应用上线后的安全性,对应用进行了加固处理,套了一层壳之后,应用又增大了1.7M,最后上线之前包的体积已经超过10M,计算器这一工具类的应用包体积超过10M就没有很强的竞争力了,所以减小包体积的工作势在必行。

下面是博主实际运用到的一些方法,没有囊括所有的方案,但都是有效的方案:

1).删除项目中无用的资源

Menu - >Refactor ->Remove unused resources

AS搜索一段时间之后会在下方弹出提示框,点击“do refactor”就可以将项目中无用的资源文件全部清除,包括图片,各种xml文件。注意:机器没有智能到那种程度,所以直接删除可能会导致一些错误,使得之后编译不通过,所以建议在直接删除之前使用快捷键“alt + F7”检查该资源文件在项目中使用的地方,确定确实无用之后在进行删除。

2).清除项目中无用的import包

Menu - >Code ->Optimize Imports  | 项目根目录下右键->Optimize Imports

以上是直接去掉整个项目中的import包,若在某个文件中去除可使用快捷键“ctrl+alt+o”。

也可在设置里直接设置为自动清除无用的import包(可自动导入包,当然也可以设置为自动删除包)。

Setting - >Editor - >General - >Auto Import - >Optimize imports on the fly

以上两步操作完之后,整个应用包减小0.7M。

3)包体分析 (Android Studio 2.2以上)- 从内到外了解

Menu - >build->Analyze APK

①lib/ -> 存放so文件,可能会有armeabi、armeabi-v7a、arm64-v8a、x86、x86_64、mips,大多数情况下只需要支持armabi与x86的架构即可,如果非必需,可以考虑拿掉x86的部分;

②res/ -> 存放编译后的资源文件,例如:drawable、layout等等;

③assets/ -> 应用程序的资源,应用程序可以使用AssetManager来检索该资源;

④META-INF/ -> 该文件夹一般存放于已经签名的APK中,它包含了APK中所有文件的签名摘要等信息;

⑤classes(n).dex -> classes文件是Java Class,被DEX编译后可供Dalvik/ART虚拟机所理解的文件格式;

从上图apk 分析后可以看到占用空间多的主要是代码、图片、资源和lib和assert文件,主要方向精简代码、压缩图片、去除无用的库、减少asserts里面文件。

AndroidManufest.xml ->最终应用请求了哪些权限,不管是主项目工程的还是外部库的,最终以打出来的apk中申请的权限为准,通过analyse apk,可以准确地知道应用到底申请了哪些权限。

有了方向之后,那么开始进行瘦身:

a.asserts文件夹内去除占用空间较大的字体文件,既使得包的体积减小(一个字体文件100-200k),又使得应用字体变得更为统一;

b.开启资源压缩

android {...buildTypes {release {minifyEnabled trueshrinkResources trueproguardFiles getDefaultProguardFile('proguard-android.txt'),'proguard-rules.pro'}}
}

minifyEnabled 设为true时会自动对代码进行清理,打包时去掉无用的代码部分,注意此时编译速度会受到影响(所以release包打包速度比debug包要慢一些);

shrinkResources设为true时打包会自动根据minifyEnabled 为true时去掉的代码部分再次去掉该代码部分相关的资源文件部分,注意二者是配合使用的。

注:build.gradle中配置二者时只是在打包文件中去掉了无用的部分,项目中还是实际存在的。

c.语言资源优化,原项目中产品经理要求配置三种语言:中文 英文 法语,最终上线前为包体积做考虑,只启用英语一门语言配置(应用是发布在谷歌应用市场)。

android {...defaultConfig {...resConfigs "en"}...
}

如果不做任何配置的话,最终APK包中会包含项目中所有已翻译语言字符串,无论应用的其余部分是否翻译为同一语言,可以通过resConfig来配置使用哪些语言,从而让构建工具移除指定语言之外的所有资源。

d.图片压缩,原本项目中使用的图片都是32*32的png图片,后来为清晰度的考虑将图片同一替换成72*72的png,结果每张图片都有200k左右,这里推荐使用tinypng网站进行图片无损压缩,下面这张图从190k压到1.6k,肉眼基本看不出任何区别;

这两张图标 一张被压缩了97%,你相信吗?

e.优化classes.dex,这里无定式方法,主要是提供一些思路 —— 时刻保持良好的编程习惯和对包体积敏锐的嗅觉,去除重复或者不用的代码,慎用第三方库,选用体积小的第三方SDK等等。

以上四步操作完之后,整个应用包减小0.5M。

4)APK瘦身放大招 - AndResGuard的使用

①概念:AndResGuard是一个缩小APK大小的工具,它的原理类似Java Proguard,但是只针对资源。它会将原本冗长的资源路径变短,例如将res/drawable/wechat变为r/d/a(是的,如同混淆一样的abc,通过进一步混淆res资源夹来减小包体积)。

②配置

项目根目录下build.gradle中,添加插件的依赖:

 dependencies {classpath 'com.tencent.mm:AndResGuard-gradle-plugin:1.2.10'}

项目根目录中build.gradle中,再添加白名单配置:

apply plugin: 'AndResGuard'
buildscript {repositories {jcenter()}dependencies {classpath 'com.android.tools.build:gradle:3.0.1'classpath 'com.tencent.mm:AndResGuard-gradle-plugin:1.2.10'}
}andResGuard {mappingFile = nulluse7zip = trueuseSign = truekeepRoot = falsewhiteList = [// 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*",]compressFilePattern = ["*.png","*.jpg","*.jpeg","*.gif","resources.arsc"]sevenzip {artifact = 'com.tencent.mm:SevenZip:1.2.10'}
}

其中whiteList(白名单)中指定不需要进行混淆的资源路径规则,主要是一些第三方SDK,因为有些SDK的代码中引用到对应的资源文件,如果对其进行混淆,会导致找不到对应资源文件,出现crash,所以不能对其资源文件进行混淆。由于公司的项目中使用到了友盟和融云,所以将这两个SDK加入白名单,更多的白名单可以查看:

AndResGuard白名单

③build.gradle文件中添加签名文件等信息

    signingConfigs {release {keyAlias 'popstar'keyPassword 'popstar'storeFile file('star.jks')storePassword 'popstar'}}buildTypes {release {minifyEnabled trueshrinkResources truesigningConfig signingConfigs.releaseproguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'}}

④打包

不再使用AS自带打包方法,而是使用gradle的打包方法,AS右侧点击Gradle打开Gradle任务列表:

打release包,只需要双击resguardRelease,开始打包(时间较长),执行完毕后,可以在app目录下的/build/output/apk/release/AndResGuard_{apk_name}/ 文件夹中找到混淆后的Apk,其中app-release_aligned_signed.apk为进行混淆并签名过的apk。

双击查看该apk,再次打开res文件夹就会发现所有的文件名都已经再次混淆了。

操作完这一步之后,整个apk减小了0.8M

有疑问的读者可以自行查看库文档 AndResGuard中文文档

 

通过以上的一些apk瘦身常用方法,最终包减小了2M左右,值得一提的是,如果使用一种瘦身方法没有明显的效果时,就可以放弃再寻找其他方法,不要死磕(别问我为什么这样说)。

PS:临时抱佛脚,应用上架前被附加了那么多东西,然后才开始做apk瘦身,博主多少有些无奈。不过学到了很多!以后不会被弄的手忙脚乱的了。

 

2020/06/02 补充:

如果按照上述步骤,使用微信压缩库时出现打包错误如下:

parse to get the exist names in the resouces.arsc first
com.tencent.mm.androlib.AndrolibException: Could not decode arsc file
at com.tencent.mm.androlib.res.decoder.RawARSCDecoder.decode(RawARSCDecoder.java:74)
at com.tencent.mm.androlib.ApkDecoder.decode(ApkDecoder.java:190)
at com.tencent.mm.resourceproguard.Main.decodeResource(Main.java:96)

原因在于升级后,系统默认升级了buildToolsVersion版本到28.0.3。从而导致对资源的解码出现问题。项目当前buildToolsVersion版本为27.0.3,直接去除,采用AGP对应的默认buildToolsVersion版本配置。

同时,AndResGuard最新版本已经解决了此问题,直接升级1.2.10版本到当前最新版本。

classpath 'com.tencent.mm:AndResGuard-gradle-plugin:1.2.17'

博主亲测,解决问题。

解决办法取自文章《项目Gradle版本从4.4升级到4.6》

这篇关于2M-APK瘦身实测可行又便捷的方法 (应用上架 阵前磨枪)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/508449

相关文章

Maven 配置中的 <mirror>绕过 HTTP 阻断机制的方法

《Maven配置中的<mirror>绕过HTTP阻断机制的方法》:本文主要介绍Maven配置中的<mirror>绕过HTTP阻断机制的方法,本文给大家分享问题原因及解决方案,感兴趣的朋友一... 目录一、问题场景:升级 Maven 后构建失败二、解决方案:通过 <mirror> 配置覆盖默认行为1. 配置示

SpringBoot排查和解决JSON解析错误(400 Bad Request)的方法

《SpringBoot排查和解决JSON解析错误(400BadRequest)的方法》在开发SpringBootRESTfulAPI时,客户端与服务端的数据交互通常使用JSON格式,然而,JSON... 目录问题背景1. 问题描述2. 错误分析解决方案1. 手动重新输入jsON2. 使用工具清理JSON3.

使用jenv工具管理多个JDK版本的方法步骤

《使用jenv工具管理多个JDK版本的方法步骤》jenv是一个开源的Java环境管理工具,旨在帮助开发者在同一台机器上轻松管理和切换多个Java版本,:本文主要介绍使用jenv工具管理多个JD... 目录一、jenv到底是干啥的?二、jenv的核心功能(一)管理多个Java版本(二)支持插件扩展(三)环境隔

Java中Map.Entry()含义及方法使用代码

《Java中Map.Entry()含义及方法使用代码》:本文主要介绍Java中Map.Entry()含义及方法使用的相关资料,Map.Entry是Java中Map的静态内部接口,用于表示键值对,其... 目录前言 Map.Entry作用核心方法常见使用场景1. 遍历 Map 的所有键值对2. 直接修改 Ma

Mybatis Plus Join使用方法示例详解

《MybatisPlusJoin使用方法示例详解》:本文主要介绍MybatisPlusJoin使用方法示例详解,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,... 目录1、pom文件2、yaml配置文件3、分页插件4、示例代码:5、测试代码6、和PageHelper结合6

Java中实现线程的创建和启动的方法

《Java中实现线程的创建和启动的方法》在Java中,实现线程的创建和启动是两个不同但紧密相关的概念,理解为什么要启动线程(调用start()方法)而非直接调用run()方法,是掌握多线程编程的关键,... 目录1. 线程的生命周期2. start() vs run() 的本质区别3. 为什么必须通过 st

C#之List集合去重复对象的实现方法

《C#之List集合去重复对象的实现方法》:本文主要介绍C#之List集合去重复对象的实现方法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录C# List集合去重复对象方法1、测试数据2、测试数据3、知识点补充总结C# List集合去重复对象方法1、测试数据

SpringBoot读取ZooKeeper(ZK)属性的方法实现

《SpringBoot读取ZooKeeper(ZK)属性的方法实现》本文主要介绍了SpringBoot读取ZooKeeper(ZK)属性的方法实现,强调使用@ConfigurationProperti... 目录1. 在配置文件中定义 ZK 属性application.propertiesapplicati

MyBatis设计SQL返回布尔值(Boolean)的常见方法

《MyBatis设计SQL返回布尔值(Boolean)的常见方法》这篇文章主要为大家详细介绍了MyBatis设计SQL返回布尔值(Boolean)的几种常见方法,文中的示例代码讲解详细,感兴趣的小伙伴... 目录方案一:使用COUNT查询存在性(推荐)方案二:条件表达式直接返回布尔方案三:存在性检查(EXI

Java调用C#动态库的三种方法详解

《Java调用C#动态库的三种方法详解》在这个多语言编程的时代,Java和C#就像两位才华横溢的舞者,各自在不同的舞台上展现着独特的魅力,然而,当它们携手合作时,又会碰撞出怎样绚丽的火花呢?今天,我们... 目录方法1:C++/CLI搭建桥梁——Java ↔ C# 的“翻译官”步骤1:创建C#类库(.NET