Android C++ Native 内存泄露检查工具使用介绍
- 实现原理
- 使用方法
- Raphael添加到测试apk
- 添加项目依赖
- 同步gradle
- 启动泄露检测功能
- 直接使用boardcast功能控制
- 在测试代码中启动
- 生成泄露报告
- 调用打印功
- 分析报告
- 分析 maps
- 停止监控
- 参考
MemoryLeakDetector 是西瓜视频基础技术团队开发的一款 native 内存泄漏监控工具,具有接入简单、监控范围广、性能优良、 稳定性好的特点。广泛用于字节跳动旗下各大 App 的 native 内存泄漏治理,收益显著!
实现原理
使用方法
步骤如下:
- 添加Raphael到要检测的apk,编译运行
- 启动泄露检测监控
- 生成泄露报告:打印堆栈内容到文件
- 使用工具泄露数据文件和mmap文件
Raphael添加到测试apk
添加项目依赖
在Android studio的工程根目录下的build.gradle中的allprojects添加:maven { url ‘https://jitpack.io’ },如下图:
allprojects {
repositories {
google()
jcenter()
flatDir {
dirs 'libs'
}
maven { url 'https://jitpack.io' }
}
}
添加依赖项目到测试工程的app目录下的build.gradle中添加:
implementation ‘com.github.bytedance:memory-leak-detector:0.2.1’
dependencies {
implementation 'com.github.bytedance:memory-leak-detector:0.2.1'
}
同步gradle
点击工具栏中的Sync project with gradle files 同步按钮
启动泄露检测功能
有2种方法启动,使用其中的一种即可:
- 使用Android的boardcast
- 添加到代码中
直接使用boardcast功能控制
监控指定so:
## 通过本地广播监控指定的so
## 0x0CF0400=Raphael.MAP64_MODE|Raphael.ALLOC_MODE|0x0F0000|1024
adb shell am broadcast -a com.bytedance.raphael.ACTION_START -f 0x01000000 --es configs 0xCF0400 --es regex ".*libXXX\\.so$"
监控进程:
## 监控整个进程(RaphaelReceiver 组件所在的进程)
## 0x0CF0400=Raphael.MAP64_MODE|Raphael.ALLOC_MODE|0x0F0000|1024
adb shell am broadcast -a com.bytedance.raphael.ACTION_START -f 0x01000000 --es configs 0xCF0400
在测试代码中启动
本步骤是可选步骤
添加Raphael.start到测试工程的kotlin/java代码里面。使用Android Studio自动import功能补全引用。
参数、条件说明:
- 需要设备位置权限,比如sdcard的读写权限
- 可以指定具体名称如libdemo.so
class MyApplication : Application() {
override fun onCreate() {
// Todo: use clientID and businessID get from xiaodu
EdgeManager.init(this, "testClient", 3001);
Log.d(TAG, "edgeSDK:" + EdgeManager.getVersion())
Raphael.start(
Raphael.MAP64_MODE or Raphael.ALLOC_MODE or 0x0F0000 or 1024,
"/storage/emulated/0/raphael", // need sdcard permission
".*libdemo.so$"
)
super.onCreate()
}
}
生成泄露报告
只有调用打印功能或者stop功能才生成泄露报告到指定目录
调用打印功
## broadcast command
adb shell am broadcast -a com.bytedance.raphael.ACTION_PRINT -f 0x01000000
分析报告
## 聚合 report,该文件在 print/stop 之后生成,需要手动 pull 出来
## 用到离线符号符号化功能的,需将raphael.py里的addr2line改为自己本地的NDK路径
## -r: 日志路径, 必需,手机端生成的report文件
## -o: 输出文件名,非必需,默认为 leak-doubts.txt
## -s: 符号表目录,非必需,有符号化需求时可传,符号表文件需跟so同名,如:libXXX.so,多个文件需放在同一目录下儿
python3 library/src/main/python/raphael.py -r report -o leak-doubts.txt -s ./symbol/
## 数据格式说明
## 201,852,591 totals // 单指raphael拦截到的未释放的虚拟内存总和
## 118,212,424 libandroid_runtime.so
## 28,822,002 libhwui.so
## 24,145,920 libstagefright.so
## 15,679,488 libv8.cr.so
## 9,566,192 libc++_shared.so
## 25,536 libsqlite.so
## 12,288 libv8_libbase.cr.so
## 5,388,741 extras // raphael.py里预设了一些通用配置,可以通过修改规则进一步识别分组到extras里的数据
##
##
## bdb11000, 70828032, 66 => bdb11000是report里此堆栈第一次分配出的内存地址,70828032是report里此堆栈的内存总和,66是report里此堆栈的总次数
## 0x000656cf /system/lib/libc.so (pthread_create + 246)
## 0x0037c129 /system/lib/libart.so (art::Thread::CreateNativeThread(_JNIEnv*, _jobject*, unsigned int, bool) + 448)
## 0x00112137 /system/framework/arm/boot.oat (java.lang.Thread.nativeCreate + 142)
分析 maps
## 分析 maps
## -m: maps文件路径,必需
python3 library/src/main/python/mmap.py -m maps
停止监控
## 广播控制
adb shell am broadcast -a com.bytedance.raphael.ACTION_STOP -f 0x01000000