Android 性能优化总结:包体积优化

news2025/1/13 17:46:17

前言

  随着开发不断迭代,App体积越来越大,包大小的增大也会给我们应用带来其他的影响 比如

  1. 下载率影响 过大的包体积会影响下载转化率,根据Google Play Store包体积和转化率分析报告显示,平均每增加1M,转化率下降0.2%左右
  2. 渠道限制 部分厂商预装强制要求安装包大小(比如国内市场在下载较大安装包就会提醒大流量是否继续下载的弹窗)
  3. 性能影响 过大的包体积在安装耗时和运行内存占用方面都会有很大影响

  但是包大小的优化不是一次就可以搞定的,需要持续的维护做好打持久战的准备,此篇文章算是在本地生活对包大小实战和别人的经验的总结,日常学习记录仅做参考。

基础了解

  在包体积优化前需要对APK做一个基本的了解

目录内容
lib文件夹下主要存放不同的cpu架构的so文件,会有armeabi、armeabi-v7a、arm64-v8a、x86、x86_64、mips,大多数情况下只需要支持armabi与x86的架构即可
res文件夹下存放编译后的资源文件,drawable和layout资源
assets应用程序的资源、字体、音频文件等,需要通过AssetManager类来访问内容
dexAndroid项目中的代码在编译后会生成.class文件,然后再通过dx工具转换为字节码文件,就是这个dex文件。一般情况下只有一个classes.dex,如果项目代码方法数超过了65535而采用了multidex的话,会有其他的.dex文件
META-INF签名信息,用来验证apk文件的完整性、合法性
resources.arsc二进制资源文件、AndroidManifest.xml清单文件

打包简要流程

image.png

打包主要有以下几步:

  1. 使用aapt工具处理所有的资源,生成一个R.java文件,一个resources.arsc文件以及其他资源。
  2. 处理.aidl文件,生成对应的Java接口文件。
  3. 将上述两步得到的R.java文件、Java接口文件,与Andorid源码一起,通过Java编译器,编译得到Java字节码文件.class文件。
  4. 获取依赖的第三方库文件,将其与上一步得到的.class文件一起,通过使用dx工具,生成.dex文件。
  5. 将资源索引文件resources.arsc、资源目录res、与上一步得到的.dex文件一起,通过apkbuilder工具,构建出初始的.apk文件。
  6. 使用jarsigner工具,对.apk文件进行签名。
  7. 使用zipalign工具,对.apk文件进行对齐。(让资源按四字节的边界进行对齐,加快资源的访问速度)

经过以上七步,一个完整的apk文件就诞生了。

那么针对压缩文件,主要的压缩体积方式分为:减少和压缩

包现状分析

使用AppChecker分析

  包体分析主要借助的是腾讯AppChecker完成的,AppChecker分析包文件主要还是借助了andoid build-tool下面的 aapt工具

image.png

  上图只是查看了各个文件类型占比,还支持统计 APK 中包含的 R 类、检查是否有多个动态库静态链接了 STL 、搜索 APK 中包含的无用资源、重复资源分析以及支持自定义检查规则等 (强烈推荐的检测工具)

借助AS提供的Analyze APK

image.png

  可以直观的查看APK的组成大小占比等信息,也可以用来查看其他产品使用了那些三方库等信息。

常规优化方式

Lint自动检测

// 扫描res资源文件
Analyze > Run Inspection by Name > Unused resources
// 扫描无用代码
Analyze > Inspect code

  不过需要注意这里扫描为静态扫描,部分资源可能存在动态调用,再删除的时候需要再三确认 ,Inspect code扫描出来的一样为静态扫描结果,反射和动态引用的代码是不会出现在这里的。

常规资源压缩

  这里主要以图片资源来进行压缩优化,列举常用方案前,简单说下各个图片文件及其特征
jpg: 一种有损的基于直接色的图片格式 属于光栅类型,所以可以表示2的24次方种颜色,非常适合色彩丰富图片、渐变色,所以相对的 jpg图片文件大小较大。

png: 也是属于光栅类型,无损压缩格式的基于8为索引色的位图格式称为png-8,支持透明度 并且文件尺寸相比jpg更小。还有一种png-24则是基于直接色的位图格式,图片存储相对较大,但是可以展示比较丰富的图像色彩。

gif: 光栅格式的图像文件类型。它使用无损压缩,但将图像“限制”为每像素 8 位和 256 色的有限调色板,常用于动画图像。

svg: 矢量图像文件类型,使用笛卡尔平面上的线和曲线系统,与总面积相比,而不是任何单个像素,可以无限放大而不会失真

Webp: 供更好的无损和有损图像压缩而开发的图像格式,相较于png 格式还可以压缩 10% -30%大小并且可以获得相同的质量

   主要有两种资源优化手段 :png文件压缩和替换为 Webp图片。

  1. png压缩,平时主要使用 https://tinypng.com/ ,不过很多UI平台都会在切图上传的时候进行自动压缩。
  2. 转为Webp, Android Studio可以右键资源文件 Convert to Webp 一键切换。

代码混淆

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

压缩( -dontshrink 关闭压缩):默认开启,用以减小应用体积,移除未被使用的类和成员,并且会在优化动作执行之后再次执行(因为优化后可能会再次暴露一些未被使用的类和成员)。

优化( -dontoptimize ):默认开启,在字节码级别执行优化,让应用运行的更快。

-dontoptimize  关闭优化
-optimizationpasses n 表示proguard对代码进行迭代优化的次数,Android一般为5

混淆( -dontobfuscate 关闭混淆):默认开启,增大反编译难度,类和类成员会被随机命名,除非用keep保护。

D8 R8优化

  Android Studio 3.1 或之后的版本 D8 将会被作为默认的 Dex 编译器,相较于D8,重点说下R8.

R8 官网地址

  根据官方说法,R8 是 Proguard 压缩与优化部分的替代品,并且它仍然使用与 Proguard 一样的 keep 规则。如果我们仅仅想在 Android Studio 中使用 R8,在 build.gradle 中打开混淆的时候,R8 就已经默认集成进 AGP 中了。

那么,R8 与混淆相比优势在哪里呢?

ProGuardR8 都应用了基本名称混淆:它们 都使用简短,无意义的名称重命名类,字段和方法。他们还可以 删除调试属性。但是,R8 在 inline 内联容器类中更有效,并且在删除未使用的类,字段和方法上则更具侵略性。例如,R8 本身集成在 ProGuard V6.1.1 版本中,在压缩 apk 的大小方面,与 ProGuard8.5% 相比,使用 R8 apk 尺寸减小了约 10%。并且,随着 Kotlin 现在成为 Android 的第一语言,R8 进行了 ProGuard 尚未提供的一些 Kotlin 的特定的优化。

从表面上看,ProGuardR8 非常相似。它们都使用相同的配置,因此在它们之间进行切换很容易。放大来看的话,它们之间也存在一些差异。R8 能更好地内联容器类,从而避免了对象分配。但是 ProGuard 也有其自身的优势,具体有如下几点:

  • ProGuard 在将枚举类型简化为原始整数方面会更加强大。它还传递常量方法参数,这通常对于使用应用程序的特定设置调用的通用库很有用。ProGuard 的多次优化遍历通常可以产生一系列优化。例如,第一遍可以传递一个常量方法参数,以便下一遍可以删除该参数并进一步传递该值。删除日志代码时,多次传递的效果尤其明显。ProGuard 在删除所有跟踪(包括组成日志消息的字符串操作)方面更有效
  • ProGuard 中应用的模式匹配算法可以识别和替换短指令序列,从而提高代码效率并为更多优化打开了机会。在优化遍历的顺序中,尤其是数学运算和字符串运算可从中受益
  • ProGuard 具有独特的能力来优化使用 GSON 库将对象序列化或反序列化为 JSON 的代码。该库严重依赖反射,这很方便,但效率低下。而 ProGuard 的优化功能可以 通过更高效,直接的访问方式 来代替它。

重复资源过滤

  由于大项目大多采用组件或插件化,多个模块之间可能存在资源的重复引入,常见的解决方法是通过资源包中的每个ZipEntry的CRC-32 checksum来筛选出重复的资源;通过android-chunk-utils修改resources.arsc,把这些重复的资源都重定向到同一个文件上; 把其它重复的资源文件从资源包中删除。

根据美团的方案之前实践过,并没有落地 实现方式较复杂,并且收益率不太高,本人并没有落地到项目中,有落地的同学可以反馈下实际优化提升。

ENUM减少使用

  如可以在开发过程 尽量减少 enum 的使用,每减少一个 enum 可以减少大约 1.0 到 1.4 KB 的大小

so文件移除

  市面上的手机cpu大多是arm架构的,所以保留arm的一种即可(定制的除外),armeabi-v7aarmeabi都可,其他直接删除。

复制代码
android {
    defaultConfig {
        ndk {
            abiFilters 'armeabi-v7a'
        }
    }
}

  在模拟器调试,就加上x86的架构,在local.properties中变量控制,正式包移除即可

  还有一种情况是不同的第三方库中存在相同的so文件
pickFirst只会打包第一个遇到的冲突的so,merge(碰到冲突会合并)和exclude(直接排除匹配到的文件,不建议使用)

packagingOptions {
       pickFirst 'lib/arm64-v8a/libgnustl_shared.so'
       pickFirst 'lib/armeabi-v7a/libgnustl_shared.so'
}

三方库处理

  实际项目开发中,各个模块都会有不同的移动团队开发,那么就可能存在重复引用的情况,比如某些不同的三方库,可能底层存在依赖同一套库的代码,那么在依赖的时候就可以进行去除

implementation('com.allenliu.versionchecklib:library:2.0.5') {
      exclude group: 'com.android.support', module: 'appcompat-v7'
      exclude group: 'com.android.support.constraint', module: 'constraint-layout'
      exclude group: 'org.greenrobot', module: 'eventbus'
      exclude group: 'com.squareup.okhttp3', module: 'okhttp'
}

三方库整合

  比如RN中使用的图片加载库是 Fresco ,但是Native使用的Glide ,那么我们就可以通过整合此类Case 达到缩减包大小目的。

  还有一种场景 比如一个三方库我们只使用一部分代码,完全可以拉出来魔改下,引入到自己的工具库中,减少多余代码引入。

进阶优化手段

插件化

   依赖于插件化的特点,将整个App拆分为多个模块,每个模块可单独运行 都是APK,最终打包的时候将宿主Apk和插件Apk分开打包,发布的时候只需要发布宿主Apk即可,用户进入不同场景按需动态下载对应Apk即可,可以很大程度上优化包体积问题。

重复技术方案筛选

  如果项目中由于历史包袱,存在多个跨端方案,比如存在 Web、小程序和 Flutter ,如果业务允许,可以将多余跨端方案移除,相对应的引擎So文件即可进行删除,起到减少包体积目的。

So动态化下载

  以Flutter为例,考虑到引擎加载和初始化时间,项目中首页一般还是采取Native方式展示,跨端场景大多在二级页面或非首页场景,按照按需思想,Flutter So也可以不放在本地,我们可以在进入跨端页面路由前,或进入App闲时阶段,添加动态下载So逻辑,相同思想,比如一些音视频文件非启动必须得话,也可以按需下载加载,本地尽量不放大文件。

DebugItem

  JVM 运行时加载的是 .class 文件,而 Android 为了使包大小更加紧凑、运行时更加高效就发明了 Dalvik 和 ART 虚拟机,两种虚拟机运行的都是 .dex 文件,当然 ART 虚拟机还可以同时运行 oat 文件。

  所以 Dex 文件里的信息内容和 Class 文件包含的信息是一样的,不同的是 Dex 文件对 Class 中的信息做了去重,一个 Dex 包含了很多的 Class 文件,并且在结构上有比较大的差异,Class 是流式的结构,Dex 是分区结构,Dex 内部的各个区块间通过 offset 来进行索引。

  为了在应用出现问题时,我们能在调试的时候去显示相应的调试信息或者上报 crash 或者主动获取调用堆栈的时候能通过 debugItem 来获取对应的行号,我们都会在混淆配置中加上下面的规则:

-keepattributes SourceFile, LineNumberTable

这样就会保留 Dex 中的 debug 与行号信息。根据 Google 官方的数据,debugItem 一般占 Dex 的比例有 5% 左右

ReDex

  ReDex 是 Facebook 开发的一个 Android 字节码的优化工具。它提供了 .dex 文件的读写和分析框架,并提供一组优化策略来提升字节码。官方提供预期优化效果:对dex文件优化为 8%

后期维稳

  在打包发布环节,可以编写插件针对各个模块大小或资源文件扫描,设置规则,写进流程中,可以防止包大小爆发式增长,当然好的包大小资源压缩思维,也可以帮助Apk维持在一个健康的水位上。

思考

   以上记录的种种方式,可根据项目状态进行选择优化,抓大放小,实事求是,也不能一味的以包大小越小越好,不影响业务,避免引起线上事故需要放在包大小之前衡量,避免得不偿失。

参考文章

美团 Android App包瘦身优化实践

jsonChao Android包体积优化

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1417211.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【MATLAB第95期】#源码分享 | 基于MATLAB的卷积神经网络CNN图像分类源代码分享(含两个案例)

【MATLAB第95期】#源码分享 | 基于MATLAB的卷积神经网络CNN图像分类源代码分享(含两个案例) 一、案例一 1、背景介绍 目的:训练和测试卷积神经网络,以检测钻头三种类型。 深度学习(DL)是机器学习的一个子…

Element table组件内容\n换行

漂亮的页面总是让人心旷神怡,层次清晰的页面让用户操作起来也是易于上手及展示。 如下的页面展示就是非常low的:用户根本阅读其中的数据。 在这个页面,根据用户填写过程生成多次填写记录,如果不进行层次性的展示,数据…

【C语言】学生管理系统

学生管理系统是一个用于管理学生信息、成绩、课程等数据的软件系统。在本文中,我们将使用C语言来实现一个简易的学生管理系统,包括学生信息的录入、显示、查询等功能。我们将使用文件来存储学生信息,以便实现持久化存储。 该学生管理…

JAVA 学习 面试(十一)常见设计模式

设计模式 ## 1、创建型模式 对象实例化的模式,创建型模式用于解耦对象的实例化过程。 单例模式:某个类智能有一个实例,提供一个全局的访问点。 工厂模式:一个工厂类根据传入的参量决定创建出哪一种产品类的实例。 抽象工厂模式&a…

Android学习之路(25) Theme和Style

1、官方详细解读 样式和主题背景 | Android 开发者 | Android Developers 2、应用场景 类似web设计中css样式。将应用设计的细节与界面的结构和行为分开。 样式style :应用于 单个 View 的外观。样式可以指定字体颜色、字号、背景颜色等属性 主题theme&…

OpenHarmony—不支持解构赋值

规则:arkts-no-destruct-assignment 级别:错误 ArkTS不支持解构赋值。可使用其他替代方法,例如,使用临时变量。 TypeScript let [one, two] [1, 2]; // 此处需要分号 [one, two] [two, one];let head, tail [head, ...tail]…

WordPress如何使用SQL实现一键关闭/开启评论功能(已有评论)

WordPress本人就自带评论功能,不过由于种种原因,有些站长不想开启评论功能,那么应该怎么实现一键关闭评论功能或开启评论功能呢?或者针对已有评论功能的文章进行一键关闭或开启评论功能应该怎么操作? 如果你使用的Wor…

每日一道面试题:Java中序列化与反序列化

写在开头 哈喽大家好,在高铁上码字的感觉是真不爽啊,小桌板又拥挤,旁边的小朋友也比较的吵闹,影响思绪,但这丝毫不影响咱学习的劲头!哈哈哈,在这喧哗的车厢中,思考着这样的一个问题…

Spring - 基本用法参考

Spring 官方文档 Spring容器启动流程(源码解读) BeanFactoryPostProcessor vs BeanPostProcessor vs BeanDefinitionRegistryPostProcessor: From java doc: BeanFactoryPostProcessor may interact with and modify bean defin…

MyBatis 如何整合 Druid 连接池?

Mybatis 如何整合 Druid 数据连接池呢&#xff1f;首先打开创建的 Maven 工程&#xff0c;找到 pom.xml 文件&#xff0c;添加 Druid 依赖。 <!--druid连接池--> <dependency><groupId>com.alibaba</groupId><artifactId>druid</artifactId&…

Mac思维导图软件XMind for mac 中文版

XMind for Mac是一款高效、易于使用的思维导图软件&#xff0c;能够帮助用户更好地组织思维和创意。无论您需要制作工作报告、演讲稿还是学习笔记&#xff0c;XMind都能够为您提供卓越的支持和帮助。 软件下载&#xff1a;XMind for mac 中文版下载 XMind for Mac具有直观的界面…

精通Python第18篇—数据之美:Pyecharts水球图绘制与交互的完整教程

Pyecharts水球图绘制与交互的完整教程 在数据可视化领域&#xff0c;Pyecharts是一个强大而灵活的工具&#xff0c;它能够以美观的方式呈现各种图表&#xff0c;其中之一就是炫酷水球图。水球图能够生动地展示数据的比例关系&#xff0c;给用户一种直观的感受。本文将深入介绍…

爬虫基础-计算机网络协议

一个数据的传输 这些设备的数据转发是通过协议来完成的&#xff0c;整个互联网可以说是完全由网络协议来维持的 不同的协议分工不同&#xff0c;比如ip协议确保了ip寻址&#xff0c;tcp协议确保了数据完整性 IP地址和URL ip地址 整个网络传输可以比作快递&#xff0c;数据就…

C51 单片机学习(一):基础外设

参考 51单片机入门教程 1. 单片机简介 1.1 定义 单片机&#xff08;Micro Controller Unit&#xff0c;简称 MCU&#xff09; 内部集成了 CPU、RAM、ROM、定时器、中断系统、通讯接口等一系列电脑的常用硬件功能单片机的任务是信息采集&#xff08;依靠传感器&#xff09;、处…

【Spark系列2】Spark编程模型RDD

RDD概述 RDD最初的概述来源于一片论文-伯克利实验室的Resilient Distributed Datasets&#xff1a;A Fault-Tolerant Abstraction for In-Memory Cluster Computing。这篇论文奠定了RDD基本功能的思想 RDD实际为Resilient Distribution Datasets的简称&#xff0c;意为弹性分…

Linux提权:Docker组挂载 Rsync未授权 Sudo-CVE Polkit-CVE

目录 Rsync未授权访问 docker组挂载 Sudo-CVE漏洞 Polkit-CVE漏洞 这里的提权手法是需要有一个普通用户的权限&#xff0c;一般情况下取得的webshell权限可能不够 Rsync未授权访问 Rsync是linux下一款数据备份工具&#xff0c;默认开启873端口 https://vulhub.org/#/envir…

Linux:共享内存

文章目录 System V共享内存的原理管理共享内存shmgetshmatshmdtshmctl 共享内存和管道实现进程间同步通信 前面介绍完了匿名管道和命名管道&#xff0c;那么本篇要引入的主题是共享内存 System V 作为进程通信部分的内容&#xff0c;共享内存必然有其存在的意义和价值&#x…

RabbitMQ快速实战

目录 什么是消息队列&#xff1f; 消息队列的优势 应用解耦 异步提速 削峰填谷 总结 主流MQ产品特点比较 Rabbitmq快速上手 创建用户admin Exchange和Queue Connection和Channel RabbitMQ中的核心概念总结 什么是消息队列&#xff1f; MQ全称Message Queue&#xf…

从零学习Linux操作系统 第二十二部分 企业域名解析服务的部署及安全优化

# 一、dns的主要信息 关于dns的名词解释&#xff1a;dns: domain name service(域名解析服务) 关于客户端: /etc/resolv.conf dns指向文件 A记录 ##ip地址叫做域名的Address 记录 SOA ##授权起始主机 关于服务端 bind安装包named服务名称/etc/named.conf主配置文件/var/na…

【深度学习:多关节嵌入模型】 Meta 解释的 ImageBind 多关节嵌入模型

【深度学习&#xff1a;多关节嵌入模型】 Meta 解释的 ImageBind 多关节嵌入模型 Meta 发布开源人工智能工具的历史分段任何模型DINOv2 什么是多模态学习&#xff1f;什么是嵌入&#xff1f;什么是 ImageBind&#xff1f;集成在 ImageBind 中的模式图像绑定架构特定模式编码器跨…