从LeakCanary看如何生成内存快照

news2025/1/9 16:36:26

前面我们已经完成了生命周期监控并且可以通过ReferenceQueue和WeakHashMap的比较确定哪些对象发生泄漏了,那么接下来需要考虑的就是如何确定这个对象是被谁持有导致泄漏的呢?

从内存泄漏一文中可知,当我们使用Android Studio或MAT分析内存泄漏问题时,通常首先是需要抓取发生内存泄漏时的hprof文件,随后使用分析软件打开该文件,排查泄漏对象被持有的GC Roots,进而解决内存泄漏问题。

同样的,在上述场景中,我们确定了发生内存泄漏的对象,那么接下来自然也是抓取此时的hprof文件,随后使用某些工具库解析该文件,进而查找发生内存泄漏对象的持有关系,找到GC Roots即可。

hprof文件

文件后缀hprof是英文Heap Profile(内存快照)的缩写,该文件是由Java虚拟机(JVM)生成的一种二进制文件,用于记录Java程序运行时内存的使用情况、对象实例分布、方法调用情况、线程统计信息等。

hprof文件通常用于分析Java程序的性能问题,查找内存泄露、对象实例分布情况、方法调用次数等,从而提高程序的性能和稳定性。

hprof文件组成

相对于Java程序生成的hprof文件而言,Android程序生成的hprof文件新增了部分TAG,Android上的hprof文件由文件头和数据区组成,其中文件头占32位,主要包含格式名和版本号,标识符大小,高位时间戳等信息,数据区由若干个Record数据组成,每个Record数据中包含Tag,数据长度,数据内容等信息,如下图所示(更多hprof文件格式可以参考:hprof文件说明):

hprof structor

生成hprof文件

在常用开发工具中都可以生成hprof文件,如Android Studio Profiler,DDMS,eclipse,命令行工具等。通过ADB生成hprof文件的命令如下所示:

adb shell am dumpheap pid/pkgName fileName.hprof

以包名com.poseidon.wanandroid为例,导出该应用当前内存快照的操作如下图所示:

image-20230817220756084

image-20230817220925949

当然也可以通过代码来生成hprof文件,在android.os.Debug类的dumpHprofData方法即可生成hprof文件,函数声明如下:

image-20230813222719042

这两个函数的区别在于:只包含fileName参数的函数是将hprof数据写入fileName对应的文件中,而后者是写入fd参数中,后者主要用于支撑前面提到的shell命令。

接前文,我们已经知道了某个对象的销毁时机,并且知道可以通过ReferenceQueue检查其是否回收,那么检查回收的机制就可以设计成在对象销毁时触发延时任务去比对ReferenceQueue和WeakHashMap,如果比对后对象的key仍在HashMap中,则发生了内存泄漏,调用Debug.dumpHprofData方法生成hprof文件。

修改ObjectWatcher代码如下所示:

public void watch(Object object) {
    removeWeaklyReachableObjects();
    String key = UUID.randomUUID().toString();
    Log.d(TAG, "watch object:" + object + ",key:" + key);
    KeyedWeakReference weakReference = new KeyedWeakReference(key, object, mReferenceQueue);
    mReferences.put(key, weakReference);

    Handler mainHandler = new Handler(Looper.getMainLooper());
    mainHandler.postDelayed(new Runnable() {
        @Override
        public void run() {
            removeWeaklyReachableObjects();
            KeyedWeakReference keyedWeakReference = mReferences.get(key);
            if (keyedWeakReference != null) {
                // key对应的object可能发生内存泄漏,dump内存堆栈
                Log.d(TAG, "keyedReference:" + keyedWeakReference.toString());
                try {
                    Debug.dumpHprofData("/sdcard/Download/tmp.hprof");
                } catch (IOException e) {
                    throw new RuntimeException(e);
                }
            }

        }
    }, 5000);
}

private void removeWeaklyReachableObjects() {
    KeyedWeakReference keyedWeakReference = null;
    do {
        keyedWeakReference = (KeyedWeakReference) mReferenceQueue.poll();
        Log.d(TAG, "keyedWeakReference:" + keyedWeakReference);
        if (keyedWeakReference != null) {
            mReferences.remove(keyedWeakReference.getKey());
            Log.d(TAG, "object has been destroyed:" + keyedWeakReference.toString());
        }
    } while (keyedWeakReference != null);
}

在MainActivity制造经典的Handler内存泄漏场景,可以看到MainActivity泄漏确实被监控到了,如下图所示:

image-20230819111121031

同时在sdcard/Download目录中也生成了prof文件,如下所示:

image-20230819105142503

使用AS打开,可以看到MainActivity的GC Roots持有路径,根据持有路径就可以分析泄漏问题并处理。

image-20230819110335652

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

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

相关文章

【从零学习python 】75. TCP协议:可靠的面向连接的传输层通信协议

文章目录 TCP协议TCP通信的三个步骤TCP特点TCP与UDP的区别TCP通信模型进阶案例 TCP协议 TCP协议,传输控制协议(英语:Transmission Control Protocol,缩写为 TCP)是一种面向连接的、可靠的、基于字节流的传输层通信协议…

java八股文面试[数据结构]——List和Set的区别

List和Set是用来存放集合的接口,并且二者都继承自接接口Collection List 中的元素存放是有序的,可以存放重复的元素,检索效率较高,插入删除效率较低。 Set 没有存放顺序不能存放重复元素检索效率较低,插入删除效率较…

【前端】深入理解CSS盒子模型与浮动

目录 一、前言二、盒子模型1、盒子模型组成1.1、border边框1.1.1、边框的三部分组成1.1.2、边框复合简写1.1.3、边框分开写1.1.4、表格的细线边框 1.2、padding内边距1.3、margin外边距1.3.1、外边距水平居中1.3.2、外边距合并1.3.3、嵌套块元素垂直 外边距的塌陷1.3.3.1、解决…

全流程R语言Meta分析核心技术应用

Meta分析是针对某一科研问题,根据明确的搜索策略、选择筛选文献标准、采用严格的评价方法,对来源不同的研究成果进行收集、合并及定量统计分析的方法,最早出现于“循证医学”,现已广泛应用于农林生态,资源环境等方面。…

LMLCCS_UPDATEFO2 LCL DB 方法 get_normvector 头寸 A 中RC 1 内部错误,过账时报错<转载>

原文链接:https://blog.csdn.net/XFYBB/article/details/129174579 物料的成本中心,作业价格没有维护 再用FCMLHELP,重新创建一下 se37,FCMLHELP_CHECK_TESTFLAG,打断点,跳过PW

外围信息收集

一、查询域名信息 1、安装whois sudo apt update sudo apt install whois2、使用 whois [域名]也可以通过在线网站进行查询网站 3、反查 4、网站在线查询 4.1、网站一 通过使用网站去查询:网址 ,这个网站只会记录他所知道的域名,不全 4.…

网络综合布线实训室方案(2023版)

综合布线实训室概述 随着智慧城市的蓬勃发展,人工智能、物联网、云计算、大数据等新兴行业也随之崛起,网络布线系统作为现代智慧城市、智慧社区、智能建筑、智能家居、智能工厂和现代服务业的基础设施和神经网络,发挥着重要作用。实践表明,网络系统故障的70%发生在布线系统,直接…

centos7 忘记密码需要重置密码

第一步进入系统加载条之前 按 e 键 第二步到了这个界面 找到 linux16 开头的 将 ro 改成 rw init/sysroot/bin/sh 修改完之后 按下按键 ctrlx 或者是 F10 第三步输入 命令 chroot /sysroot第四步 重置root用户密码 这里重置密码是叫你输入两遍密码 passwd root第五步 更…

常量变量习题答案

基础题目: 第一题 按步骤编写代码,效果如图所示: 编写步骤: 定义类 Test1定义 main方法控制台输出5行字符串类型常量值控制台输出5行字符类型常量值 参考答案: public class Test1 {public static void main(String[] args) {/…

【负载均衡】Nacos简单入门

Nacos简单入门 快速安装 在Nacos的GitHub页面,提供有下载链接,可以下载编译好的Nacos服务端或者源代码: 下载完压缩包之后,放在任意目录下面进行解压: GitHub主页:https://github.com/alibaba/nacos G…

Java日志框架概览

SLF4J 提供统一的日志门面API,即图中紫色部分,实现中立的日志记录API 桥接功能,蓝色部分,把各种日志框架API(绿色部分)桥接到SLF4J API。这样即便你的程序中使用各种日志API记录日志,最终都可桥…

伊斯特曼E1D和VEAZEN费森S88这两款全单吉他怎么样?优缺点综合对比评测,哪一款更出众?

3000-4000元价位预算有什么适合初学或者进阶换琴的吉他品牌推荐?VEAZEN费森S88系列和Eastman伊斯特曼E1D系列一直也是深受初学者和进阶者喜欢的热销全单吉他型号,那么,今天就以它们为本期的评测主角,从五个方面深度剖析介绍和对比…

计算机网络第3章(数据链路层)

计算机网络第3章(数据链路层) 3.1 数据链路层概述3.1.1 概述3.1.2 数据链路层使用的信道3.1.3 三个重要问题 3.2 封装成帧3.2.1 介绍3.2.2 透明传输3.2.3 总结 3.3 差错检测3.3.1 介绍3.3.2 奇偶校验3.3.3 循环冗余校验CRC(Cyclic Redundancy Check)3.3.…

EasyExcel+POI制作带有有效性校验及下拉联动的Excel模板

文章目录 1.背景2.实现功能的Excel特性2.1.特性介绍2.2.下拉框联动2.3.单元格自动匹配Id2.4.错误提示 3.代码实现3.1.基础流程代码3.2.名称管理器配置3.3.有效性配置3.4.函数填充3.5.其他补充 4.总结 1.背景 最近在做一个CRM系统的人员销售目标导入的相关需求,需要…

壁仞科技与百度飞桨完成II级兼容性测试

近日,壁仞科技BR104通用GPU与百度飞桨已完成II级兼容性测试。测试结果显示,双方兼容性表现良好,整体运行稳定。这是壁仞科技加入飞桨“硬件生态共创计划”后的阶段性成果。产品兼容性证明本次II级兼容性测试完成了涵盖自然语言处理、计算机视…

FFmpeg解码32k大分辨率出现如下错误:Picture size 32768x32768 is invalid

最近找到一张32k的jpeg图片,尝试用ffmpeg来进行解码,命令如下: ffmpeg -i enflame_32768-32768-420.jpg 32.yuv结果出现Picture size 32768x32768 is invalid的错误:

【C++】用Windows API在控制台实现选择选项

2023年8月23日&#xff0c;周三上午 今天上午花了一个小时来实现这个 这个程序在碰到边界时会发出声音&#xff0c; 通过调用Windows API的Beep函数来实现。 #include<Windows.h> #include<conio.h> #include<iostream> #include<cstdlib>const int …

生信豆芽菜-EMT评分的计算

网址&#xff1a;http://www.sxdyc.com/gradeEmt 1、数据准备 表达谱数据&#xff0c;行为基因&#xff0c;列为样本 2、提交后&#xff0c;等待运行成功即可下载 当然&#xff0c;如果不清楚数据是什么样的&#xff0c;可以选择下载我们的示例数据&#xff0c;也可以关注…

统计分析查找总人数、男、女的人数

SELECTcount( * ) AS count,sum( CASE WHEN MOD ( substr( remark, 17, 1 ), 2 ) 0 THEN 1 ELSE 0 END ) AS women,sum( CASE WHEN MOD ( substr( remark, 17, 1 ), 2 ) 1 THEN 1 ELSE 0 END ) AS man FROMt_user

CH583/2构建工程教程

CH583/2构建工程教程 绪论资源移植步骤准备移植步骤一步骤二 工程配置修改工程名修改前修改后 工程配置修改资源文件 修改C/C general修改C/C构建修改汇编交叉编译修改C交叉编译修改GNU RISC-V Cross Linker 修改跟编译 移植注意事项 绪论 资源 CH583/2的SDK下载 移植步骤 …