04 | 内存优化(下):内存优化这件事,应该从哪里着手?
在掌握内存相关的背景知识后,下一步你肯定想着手开始优化内存的问题了。不过在真正开始做内存优化之前,需要先评估内存对应用性能的影响,我们可以通过崩溃中“异常退出” 和 OOM 的比例进行评估。另一方面,低内存设备更容易出现内存不足引起的异常和卡顿,我们也可以通过查看应用中用户的手机内存在 2GB 以下所占的比例来评估。
所以在优化前要先定好自己的目标,这一点非常关键。比如针对 512MB 的设备和针对 2GB 以上的设备,完全是两种不同的优化思路。如果我们面向东南亚、非洲用户,那对内存优化的标准就要变得更苛刻一些。
铺垫了这么多,下面我们就来看看内存优化都有哪些方法吧。
内存优化探讨
那要进行内存优化,应该从哪里着手呢?我通常会从设备分级、Bitmap 优化和内存泄漏这三个方面入手
1. 设备分级
相信你肯定遇到过,同一个应用在 4GB 内存的手机运行得非常流畅,但在 1GB 内存的手机就不一定可以做到,而且在系统空闲和繁忙的时候表现也不太一样。
内存优化首先需要根据设备环境来综合考虑
,专栏上一期我提到过很多同学陷入的一个误区:“内存占用越少越好”。其实我们可以让高端设备使用更多的内存,做到针对设备性能的好坏使用不同的内存分配和回收策略。
当然这需要有一个良好的架构设计支撑,在架构设计时需要做到以下几点。
-
设备分级。使用类似
device-year-class
的策略对设备分级,对于低端机用户可以关闭复杂的动画,或者是某些功能;使用 565 格式的图片,使用更小的缓存内存等。在现实环境下,不是每个用户的设备都跟我们的测试机一样高端,在开发过程我们要学会思考功能要不要对低端机开启、在系统资源吃紧的时候能不能做降级。
下面我举一个例子。我们知道 device-year-class 会根据手机的内存、CPU 核心数和频率等信息决定设备属于哪一个年份,这个示例表示对于 2013 年之后的设备可以使用复杂的动画,对于 2010 年之前的低端设备则不添加任何动画。
安装包中的代码、图片、资源以及 so 库的大小跟内存究竟有哪些关系?你可以参考下面的这个表格。
安装包中代码、图片、资源、so库的大小与内存的关系
Bitmap 内存一般占应用总内存很大一部分,所以做内存优化永远无法避开图片内存这个“永恒主题”。
即使把所有的 Bitmap 都放到 Native 内存,并不代表图片内存问题就完全解决了,这样做只是提升了系统内存利用率,减少了 GC 带来的一些问题而已。
那我们回过头来看看,到底该如何优化图片内存呢?我给你介绍两种方法。
方法一,统一图片库。
图片内存优化的前提是收拢图片的调用,这样我们可以做整体的控制策略。例如低端机使用 565 格式、更加严格的缩放算法,可以使用 Glide、Fresco 或者采取自研都可以。而且需要进一步将所有 Bitmap.createBitmap、BitmapFactory 相关的接口也一并收拢。
方法二,统一监控。
在统一图片库后就非常容易监控 Bitmap 的使用情况了,这里主要有三点需要注意。
-
大图片监控。我们需要注意某张图片内存占用是否过大,例如长宽远远大于 View 甚至是屏幕的长宽。在开发过程中,如果检测到不合规的图片使用,应该立即弹出对话框提示图片所在的 Activity 和堆栈,让开发同学更快发现并解决问题。在灰度和线上环境下可以将异常信息上报到后台,我们可以计算有多少比例的图片会超过屏幕的大小,也就是图片的
“超宽率”
。
-
重复图片监控。重复图片指的是 Bitmap 的像素数据完全一致,但是有多个不同的对象存在。这个监控不需要太多的样本量,一般只在内部使用。
之前我实现过一个内存 Hprof 的分析工具,它可以自动将重复 Bitmap 的图片和引用链输出
。下图是一个简单的例子,你可以看到两张图片的内容完全一样,通过解决这张重复图片可以节省 1MB 内存。
-
图片总内存。通过收拢图片使用,我们还可以统计应用所有图片占用的内存,这样在线上就可以按不同的系统、屏幕分辨率等维度去分析图片内存的占用情况。
在 OOM 崩溃的时候,也可以把图片占用的总内存、Top N 图片的内存都写到崩溃日志中,帮助我们排查问题。
讲完设备分级和 Bitmap 优化,我们发现架构和监控需要两手抓,一个好的架构可以减少甚至避免我们犯错,而一个好的监控可以帮助我们及时发现问题。
3. 内存泄漏
内存泄漏简单来说就是没有回收不再使用的内存,排查和解决内存泄漏也是内存优化无法避开的工作之一。
内存泄漏主要分两种情况,一种是同一个对象泄漏,还有一种情况更加糟糕,就是每次都会泄漏新的对象,可能会出现几百上千个无用的对象。
很多内存泄漏都是框架设计不合理所导致,各种各样的单例满天飞,MVC 中 Controller 的生命周期远远大于 View。优秀的框架设计可以减少甚至避免程序员犯错,当然这不是一件容易的事情,所以我们还需要对内存泄漏建立持续的监控。
-
Java 内存泄漏
。建立类似 LeakCanary 自动化检测方案,至少做到 Activity 和 Fragment 的泄漏检测。在开发过程,我们希望出现泄漏时可以弹出对话框,让开发者更加容易去发现和解决问题。内存泄漏监控放到线上并不容易,我们可以对生成的 Hprof 内存快照文件做一些优化,裁剪大部分图片对应的 byte 数组减少文件大小。
比如一个 100MB 的文件裁剪后一般只剩下 30MB 左右,使用 7zip 压缩最后小于 10MB,增加了文件上传的成功率
。
-
OOM监控
。美团有一个 Android 内存泄露自动化链路分析组件
Probe
,它在发生 OOM 的时候生成 Hprof 内存快照,然后通过单独进程对这个文件做进一步的分析。不过在线上使用这个工具风险还是比较大,在崩溃的时候生成内存快照
有可能会导致二次崩溃
,而且部分手机生成 Hprof 快照可能会耗时几分钟,这对用户造成的体验影响会比较大。另外,部分 OOM 是因为虚拟内存不足导致,这块需要具体问题具体分析。
-
Native内存监控
。上一期我讲到 Malloc 调试(Malloc Debug)和 Malloc 钩子(Malloc Hook)似乎还不是那么稳定。在 WeMobileDev 最近的一篇文章《
微信 Android 终端内存优化实践
》中,微信也做了一些其他方案上面的尝试。
matrix中已经开源了采用xhook hook内存分配函数,从而达到Native内存监控目的的组件,代码位于
matrix/matrix-android/matrix-hooks/src/main/java/com/tencent/matrix/hook/memory
-
针对无法重编 so 的情况
,使用了 PLT Hook 拦截库的内存分配函数,其中 PLT Hook 是 Native Hook 的一种方案,后面我们还会讲到。然后重定向到我们自己的实现后记录分配的内存地址、大小、来源 so 库路径等信息,定期扫描分配与释放是否配对,对于不配对的分配输出我们记录的信息。
-
针对可重编的 so 情况
,通过 GCC 的“-finstrument-functions”参数给所有函数插桩,桩中模拟调用栈入栈出栈操作;通过 ld 的“–wrap”参数拦截内存分配和释放函数,重定向到我们自己的实现后记录分配的内存地址、大小、来源 so 以及插桩记录的调用栈此刻的内容,定期扫描分配与释放是否配对,对于不配对的分配输出我们记录的信息。
开发过程中内存泄漏排查可以使用 Androd Profiler 和 MAT 工具配合使用,而日常监控关键是成体系化,做到及时发现问题。
坦白地说,除了 Java 泄漏检测方案,目前 OOM 监控和 Native 内存泄漏监控都只能做到实验室自动化测试的水平。微信的 Native 监控方案也遇到一些兼容性的问题,如果想达到灰度和线上部署,需要考虑的细节会非常多。Native 内存泄漏检测在 iOS 会简单一些,不过 Google 也在一直优化 Native 内存泄漏检测的性能和易用性,相信在未来的 Android 版本将会有很大改善。
内存监控
前面我也提了内存泄漏的监控存在一些性能的问题,一般只会对内部人员和极少部分的用户开启。在线上我们需要通过其他更有效的方式去监控内存相关的问题。
1. 采集方式
用户在前台的时候,可以每 5 分钟采集一次 PSS、Java 堆、图片总内存。我建议通过采样只统计部分用户,需要注意的是要按照用户抽样,而不是按次抽样。简单来说一个用户如果命中采集,那么在一天内都要持续采集数据。
2. 计算指标
通过上面的数据,我们可以计算下面一些内存指标。
内存异常率
:可以反映内存占用的异常情况,如果出现新的内存使用不当或内存泄漏的场景,这个指标会有所上涨。其中 PSS 的值可以通过 Debug.MemoryInfo 拿到。
触顶率
:可以反映 Java 内存的使用情况,如果超过 85% 最大堆限制,GC 会变得更加频繁,容易造成 OOM 和卡顿。
其中是否触顶可以通过下面的方法计算得到。
一般客户端只上报数据,所有计算都在后台处理,这样可以做到灵活多变。后台还可以计算平均 PSS、平均 Java 内存、
平均图片占用
这些指标,它们可以反映内存的平均情况。通过平均内存和分区间内存占用这些指标,我们可以通过版本对比来监控有没有新增内存相关的问题。
因为上报了前台时间,我们还可以按照时间维度看应用内存的变化曲线。比如可以观察一下我们的应用是不是真正做到了“
用时分配,及时释放
”。如果需要,我们还可以实现按照场景来对比内存的占用。
在实验室或者内部试用环境,我们也可以通过 Debug.startAllocCounting 来监控 Java 内存分配和 GC 的情况,需要注意 的是这个选项对性能有一定的影响,虽然目前还可以使用,但已经被 Android 标记为 deprecated。
通过监控,我们可以拿到内存分配的次数和大小,以及 GC 发起次数等信息。
上面的这些信息似乎不太容易定位问题,在 Android 6.0 之后系统可以拿到更加精准的 GC 信息。
需要特别注意阻塞式 GC 的次数和耗时,因为它会暂停应用线程,可能导致应用发生卡顿。我们也可以更加细粒度地分应用场景统计,例如启动、进入朋友圈、进入聊天页面等关键场景。
在具体进行内容优化前,我们首先要问清楚自己几个问题,比如我们要优化到什么目标、内存对我们造成了多少异常和卡顿。只有在明确了应用的现状和优化目标后,我们才能去进行下一步的操作。
在探讨了内存优化的思路时,针对不同的设备、设备不同的情况,我们希望可以给用户不同的体验。这里我主要讲到了关于 Bitmap 内存优化和内存泄漏排查、监控的一些方法。最后我提到了怎样在线上监控内存的异常情况,通常内存异常率、触顶率这些指标对我们很有帮助。
目前我们在 Native 泄漏分析上做的还不是那么完善,不过做优化工作的时候,我特别喜欢用演进的思路来看问题。用演进的思路来看,即使是 Google, 在时机不成熟时也会做一些权衡和妥协。换到我们个人身上,等到时机成熟或者我们的能力达到了,就需要及时去还这些“技术债务”。
课后作业
使用HAHA库快速判断内存中是否存在重复的图片,且将这些重复图片的PNG、堆栈等信息输出。
该作业可以参考微信开源的Matrix中的部分
DuplicatedBitmapAnalyzer.java
,效果更好。
自己的作业如下,需要借助
leakcanary-analayzer-1.6.2.jar
以及
leakcanary-watcher-1.6.2.jar
解析堆栈信息。
实践起来还是有一些问题:
-
对比之下,堆栈打印不准确
-
自己作业对比Matrix的,找出来的元素更多,可能是误报
package com.ximalaya.ting.kid.bitmap
import com.google.gson.GsonBuilder
import com.squareup.haha.perflib.*
import com.squareup.haha.perflib.io.MemoryMappedFileBuffer
import com.squareup.leakcanary.*
import java.io.File
import java.io.FileOutputStream
import java.io.OutputStreamWriter
import java.util.*
* Created by yorek.liu 2020/5/14
* @author yorek.liu
* email [email protected]
private const val TAG = "DuplicateBitmapChecker"
class DuplicateBitmapChecker {
private lateinit var mHprofFilePath: String
private lateinit var mOutputDir: String
private lateinit var mOutputFilePath: String
fun init(hprofFilePath: String, outputDir: String, outputFilePath: String) {
mHprofFilePath = hprofFilePath
mOutputDir = outputDir
mOutputFilePath = outputFilePath
fun parseHprof() {
LogWrapper.i(TAG, "ready to parse hprof file: $mHprofFilePath")
val hprofFile = File(mHprofFilePath)
val buffer = MemoryMappedFileBuffer(hprofFile)
val parser = HprofParser(buffer)
val snapshot = parser.parse()
snapshot.computeDominators()
LogWrapper.i(TAG, "parse hprof file completed")
val bitmapClasses = snapshot.findClasses("android.graphics.Bitmap")
val heaps = snapshot.heaps
LogWrapper.i(TAG, "bitmapClasses.size=${bitmapClasses.size}, heaps.size=${heaps.size}")
val gson = GsonBuilder()
.setPrettyPrinting()
.disableHtmlEscaping()
.create()
val fileOutputStream = FileOutputStream(mOutputFilePath)
val fileWriter = OutputStreamWriter(fileOutputStream)
heaps.filter {
it.name == "app" || it.name == "default"
}.forEach { heap ->
analyzeHeap(bitmapClasses, heap, snapshot).forEach { analyzerResult ->
fileWriter.write(analyzerResult.toJson(gson))
fileWriter.flush()
fileWriter.close()
fileOutputStream.close()
LogWrapper.i(TAG, "save output to $mOutputFilePath")
LogWrapper.i(TAG, "save outputImages to dir: $mOutputDir")
private fun analyzeHeap(bitmapClasses: MutableCollection<ClassObj>, heap: Heap, snapshot: Snapshot): List<AnalyzerResult> {
val resultList = mutableListOf<AnalyzerResult>()
bitmapClasses.forEach { bitmapClass ->
val instances = bitmapClass.getHeapInstances(heap.id)
LogWrapper.i(TAG, "======================================================")
LogWrapper.i(TAG, "${instances.size} instances found in heap ${heap.name}")
val hashCodeList = IntArray(instances.size) { 0 }
val hashCode2InstanceMap = HashMap<Int, MutableList<Instance>>()
LogWrapper.i(TAG, "calculating hashcode ...")
for (i in instances.indices) {
if (instances[i].distanceToGcRoot == Int.MAX_VALUE) {
continue
val hashCode = getHashCode(instances[i])
hashCodeList[i] = hashCode
var instanceList = hashCode2InstanceMap[hashCode]
if (instanceList == null) {
instanceList = mutableListOf()
hashCode2InstanceMap[hashCode] = instanceList
instanceList.add(instances[i])
LogWrapper.i(TAG, "${hashCode2InstanceMap.size} bitmap instance found")
hashCode2InstanceMap.values.forEachIndexed { index, instanceList ->
if (instanceList.size == 0 || instanceList.size == 1) {
LogWrapper.i(TAG, "analyzing #$index: skip due to the size: ${instanceList.size}")
return@forEachIndexed
LogWrapper.i(TAG, "analyzing #$index ...")
val analyzerResult = getAnalyzerResult(instanceList, snapshot)
resultList.add(analyzerResult)
LogWrapper.i(TAG, "======================================================")
LogWrapper.i(TAG, "")
return resultList
private fun getHashCode(instance: Instance): Int {
val classInstanceValues = (instance as ClassInstance).values
val buffer = fieldValue<ArrayInstance>(classInstanceValues, "mBuffer")
return Arrays.hashCode(buffer.values)
private fun getAnalyzerResult(instanceList: List<Instance>, snapshot: Snapshot): AnalyzerResult {
val instance = instanceList[0]
val classInstanceValues = (instance as ClassInstance).values
val buffer = fieldValue<ArrayInstance>(classInstanceValues, "mBuffer")
val bitmapWidth = fieldValue<Int>(classInstanceValues, "mWidth")
val bitmapHeight = fieldValue<Int>(classInstanceValues, "mHeight")
val analyzerResult = AnalyzerResult()
analyzerResult.duplicateCount = instanceList.size
analyzerResult.bufferSize = buffer.size
analyzerResult.width = bitmapWidth
analyzerResult.height = bitmapHeight
try {
val method = buffer.javaClass.getDeclaredMethod("asRawByteArray", Int::class.java, Int::class.java)
method.isAccessible = true
val imageDataByteArray = method.invoke(buffer, 0, buffer.size) as ByteArray
method.isAccessible = false
analyzerResult.bufferHash = DigestUtil.getMD5String(imageDataByteArray)
analyzerResult.imageOutput = getOutputImagePath(analyzerResult.bufferHash)
ARGB8888_BitmapExtractor.outputImage(bitmapWidth, bitmapHeight, imageDataByteArray, analyzerResult.imageOutput)
} catch (e: Exception) { }
for (ins in instanceList) {
val leakTrace = getLeakTrace(ins, snapshot)
analyzerResult.stacks.add(leakTrace.parse())
return analyzerResult
private fun getLeakTrace(instance: Instance, snapshot: Snapshot): LeakTrace? {
val noOpRefs = ExcludedRefs.builder().build()
val heapAnalyzer = HeapAnalyzer(noOpRefs, AnalyzerProgressListener.NONE, emptyList())
try {
val method = heapAnalyzer.javaClass.getDeclaredMethod(
"findLeakTrace",
Long::class.java,
Snapshot::class.java,
Instance::class.java,
Boolean::class.java
method.isAccessible = true
val analyzerResult = method.invoke(
heapAnalyzer,
System.nanoTime(),
snapshot,
instance,
false
) as AnalysisResult
return analyzerResult.leakTrace
} catch (e: java.lang.Exception) {
e.printStackTrace()
return null
private fun <T> fieldValue(values: List<ClassInstance.FieldValue>, fieldName: String): T {
for (fieldValue in values) {
if (fieldValue.field.name == fieldName) {
return fieldValue.value as T
throw java.lang.IllegalArgumentException("Field $fieldName does not exists")
private fun getOutputImagePath(fileName: String): String {
return "$mOutputDir${File.separator}$fileName.png"
package com.ximalaya.ting.kid.bitmap;
import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException;
public final class DigestUtil {
private static final char[] HEX_DIGITS
= {'0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'a', 'b', 'c', 'd', 'e', 'f'};
public static String getMD5String(byte[] buffer) {
try {
final MessageDigest md = MessageDigest.getInstance("MD5");
md.update(buffer);
final byte[] resBytes = md.digest();
return bytesToHexString(resBytes);
} catch (NoSuchAlgorithmException e) {
// Should not happen.
throw new IllegalStateException(e);
private static String bytesToHexString(byte[] bytes) {
final StringBuilder sb = new StringBuilder();
for (byte b : bytes) {
if (b >= 0 && b <= 15) {
sb.append('0').append(HEX_DIGITS[b]);
} else {
sb.append(HEX_DIGITS[(b >> 4) & 0x0F]).append(HEX_DIGITS[b & 0x0F]);
return sb.toString();
private DigestUtil() {
throw new UnsupportedOperationException();