JVM-使用MAT(Memory Analyzer Tool)内存分析工具分析JVM老生代增长异常

本文主要是介绍JVM-使用MAT(Memory Analyzer Tool)内存分析工具分析JVM老生代增长异常,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

JVM-使用MAT(Memory Analyzer Tool)内存分析工具分析JVM老生代增长异常

最近通过监控发现java服务的堆内存每天都会上涨150MB左右,通过分析监控发现是堆内存中的PS Old Gen老生代)区域每天上涨导致堆内存上涨,而且也不会被回收; 本篇文章就带领大家通过MAT(Memory Analyzer Tool)内存分析工具一起找出老生代内存长涨的对象

目录

  • JVM-使用MAT(Memory Analyzer Tool)内存分析工具分析JVM老生代增长异常
  • 一、准备内容
  • 二、分析老生代内存区域中的对象
    • 2.1 获取java服务的PID
    • 2.2 Dump堆内存成.hprof文件
    • 2.3 获取老生代内存区域的内存地址范围
    • 2.4 使用MAT分析老生代内存中的对象

一、准备内容

  1. MAT(Memory Analyzer Tool)内存分析工具
    点击【官方下载】进行下载;
    在这里插入图片描述
    MAT依赖本地JAVA_HOME环境变量设置,如果本地JAVA_HOME是JDK8的话,直接将JDK11下载到MAT安装目录,并将JDK11的路径通过-vm参数添加到MemoryAnalyzer.ini配置文件中
    在这里插入图片描述

  2. JDK11 MAT依赖最低JDK11,请大家自行前往oracle官方网站下载;

  3. JVisualvm、Jmap或【阿里开源的java分析工具Arthas】;

  4. vjmap

一款开源的增强版jmap,用于输出老生代内存区的内存地址,【下载地址】
可以下载vjtools,通过maven编译vjmap,通过maven的install命令会在目标目录生成vjmap.zip,里面是完整的运行环境
在这里插入图片描述

二、分析老生代内存区域中的对象

2.1 获取java服务的PID

  1. 如果是Windows系统的话,可以通过系统的任务管理器中获取
    Windows获取java服务PID
  2. 如果是Linux系统,可以通过命令来进行查询PID
# ps -aef | grep java

如下图所示:
Linux查询java服务的PID
3. JVisualVM可以显示java服务的PID
在这里插入图片描述

备注: 以上3种方式选一种即可

2.2 Dump堆内存成.hprof文件

  1. 通过JVisualVM(Oracle JDK内置的监控工具)Dump出堆内存信息
    • 点击监控页签->[堆Dump]按钮在这里插入图片描述
    • 稍等一会就会生成hprof文件,在详情画面会显示导出路径,右击此路径可以复制在这里插入图片描述
  2. 通过Arthas工具中的heapdump命令导出堆内存中的信息
  • 启动arthas-boot.jar,输入PID对应的序号
java -jar arthas-boot.jar

如下图所示:
在这里插入图片描述

  • 执行导出堆内存
[arthas]$ heapdump

在这里插入图片描述

说明:上面是导出堆内存信息的两种方式,选一种即可,需复制下hprof文件的导出路径,MAT会用到。

2.3 获取老生代内存区域的内存地址范围

  1. 运行vjmap命令输出老生代的内存地址
vjmap.bat -address 15832 

在这里插入图片描述

Linux的运行vjmap.sh -address PID命令即可

输出信息:
PSYoungGen [
eden = [0x00000007d5560000,0x00000007db9e1478,0x00000007f3250000] ,
from = [0x00000007f3250000,0x00000007f3be8000,0x00000007f9920000] ,
to = [0x00000007f9a40000,0x00000007f9a40000,0x0000000800000000] ]
PSOldGen [ [0x0000000780000000,0x00000007846715f8,0x00000007d5560000] ]

由此可以得出老生代内存地址范围为PSOldGen [ [0x0000000780000000,0x00000007846715f8,0x00000007d5560000] ],其中
0x0000000780000000:PSOldGen的起始地址
0x00000007846715f8:表示PSOldGen已使用了的结束地址;
0x00000007d5560000:表示PSOldGen的结束地址;

2.4 使用MAT分析老生代内存中的对象

  1. 打开刚才导出的hprof文件
    在这里插入图片描述

  2. 进入OQL编辑画面,执行查询老生代内存区域中的对象
    在这里插入图片描述
    查询OQL语句如下:

SELECT * 
FROM INSTANCEOF java.lang.Object t 
WHERE toHex(t.@objectAddress)>="0x780000000" AND toHex(t.@objectAddress)<="0x7d5560000"
  • 注: 上面oql语句条件中的objectAddress中的值是通过vjmap工具输出信息中的PSOldGen得出的,记得使用时要删除0x后面的00000;
  • 上图中的shallow Heap是指对象在没有引用其它对象、数组时,对象本身的大小;
  • 上图中的Retained Heap是指对象本身的大小+它所引用对象的大小,即对象的shallow Heap+该对象所引用的所有对象大小的总和,即GC要回收该对象的内存大小

到这一步,剩下的工作就是分析上图所列出的对象,分析其GC路径和引用路径,分析程序中产生内存泄露的代码了;

这篇关于JVM-使用MAT(Memory Analyzer Tool)内存分析工具分析JVM老生代增长异常的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

一篇文章彻底搞懂macOS如何决定java环境

《一篇文章彻底搞懂macOS如何决定java环境》MacOS作为一个功能强大的操作系统,为开发者提供了丰富的开发工具和框架,下面:本文主要介绍macOS如何决定java环境的相关资料,文中通过代码... 目录方法一:使用 which命令方法二:使用 Java_home工具(Apple 官方推荐)那问题来了,

Java HashMap的底层实现原理深度解析

《JavaHashMap的底层实现原理深度解析》HashMap基于数组+链表+红黑树结构,通过哈希算法和扩容机制优化性能,负载因子与树化阈值平衡效率,是Java开发必备的高效数据结构,本文给大家介绍... 目录一、概述:HashMap的宏观结构二、核心数据结构解析1. 数组(桶数组)2. 链表节点(Node

Java AOP面向切面编程的概念和实现方式

《JavaAOP面向切面编程的概念和实现方式》AOP是面向切面编程,通过动态代理将横切关注点(如日志、事务)与核心业务逻辑分离,提升代码复用性和可维护性,本文给大家介绍JavaAOP面向切面编程的概... 目录一、AOP 是什么?二、AOP 的核心概念与实现方式核心概念实现方式三、Spring AOP 的关

详解SpringBoot+Ehcache使用示例

《详解SpringBoot+Ehcache使用示例》本文介绍了SpringBoot中配置Ehcache、自定义get/set方式,并实际使用缓存的过程,文中通过示例代码介绍的非常详细,对大家的学习或者... 目录摘要概念内存与磁盘持久化存储:配置灵活性:编码示例引入依赖:配置ehcache.XML文件:配置

Java 虚拟线程的创建与使用深度解析

《Java虚拟线程的创建与使用深度解析》虚拟线程是Java19中以预览特性形式引入,Java21起正式发布的轻量级线程,本文给大家介绍Java虚拟线程的创建与使用,感兴趣的朋友一起看看吧... 目录一、虚拟线程简介1.1 什么是虚拟线程?1.2 为什么需要虚拟线程?二、虚拟线程与平台线程对比代码对比示例:三

Nginx分布式部署流程分析

《Nginx分布式部署流程分析》文章介绍Nginx在分布式部署中的反向代理和负载均衡作用,用于分发请求、减轻服务器压力及解决session共享问题,涵盖配置方法、策略及Java项目应用,并提及分布式事... 目录分布式部署NginxJava中的代理代理分为正向代理和反向代理正向代理反向代理Nginx应用场景

k8s按需创建PV和使用PVC详解

《k8s按需创建PV和使用PVC详解》Kubernetes中,PV和PVC用于管理持久存储,StorageClass实现动态PV分配,PVC声明存储需求并绑定PV,通过kubectl验证状态,注意回收... 目录1.按需创建 PV(使用 StorageClass)创建 StorageClass2.创建 PV

Redis 基本数据类型和使用详解

《Redis基本数据类型和使用详解》String是Redis最基本的数据类型,一个键对应一个值,它的功能十分强大,可以存储字符串、整数、浮点数等多种数据格式,本文给大家介绍Redis基本数据类型和... 目录一、Redis 入门介绍二、Redis 的五大基本数据类型2.1 String 类型2.2 Hash

Java中的.close()举例详解

《Java中的.close()举例详解》.close()方法只适用于通过window.open()打开的弹出窗口,对于浏览器的主窗口,如果没有得到用户允许是不能关闭的,:本文主要介绍Java中的.... 目录当你遇到以下三种情况时,一定要记得使用 .close():用法作用举例如何判断代码中的 input

Redis中Hash从使用过程到原理说明

《Redis中Hash从使用过程到原理说明》RedisHash结构用于存储字段-值对,适合对象数据,支持HSET、HGET等命令,采用ziplist或hashtable编码,通过渐进式rehash优化... 目录一、开篇:Hash就像超市的货架二、Hash的基本使用1. 常用命令示例2. Java操作示例三