JVM 排查工具介绍(二)Memory Analyzer 堆内存分析工具
一、前提条件准备
在官网 http://www.eclipse.org/mat/downloads.php 中可以下载,下载到本地之后进行安装
二、案例
1、分析堆内存泄漏
1) 代码
package com.wf.example.jvm;
import java.util.List;
import java.util.concurrent.CopyOnWriteArrayList;
import lombok.extern.slf4j.Slf4j;
/**
* 用来测试 jvm 进行 GC之后,内存是否归还给操作系统的验证
*
* @author sandy
*
*/
@Slf4j
public class JVMDemo {
public static void main(String[] args) {
@SuppressWarnings("rawtypes")
List list = new CopyOnWriteArrayList<>();
int count = 512;
new Thread(() -> {
try {
for (int i = 0; i < 10; i++) {
log.info(String.format("第%s次生产%s大小的对象", i, count));
addObject(list, count);
Thread.sleep(i * 30000);
}
} catch (Throwable ex) {
log.error("execute fail", ex);
}
},"productThread").start();
new Thread(() -> {
while (true) {
if (list.size() >= count) {
log.info("清理List... 回收 jvm内存....");
log.info("before");
printJvmMemoryInfo();
list.clear();
System.gc();
log.info("after");
printJvmMemoryInfo();
}
}
},"cleanThread").start();
try {
Thread.currentThread().join();
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
@SuppressWarnings({ "unchecked", "rawtypes" })
private static void addObject(List list, int count) {
for (int i = 0; i < count; i++) {
log.info("now list.size[{}]",list.size());
OOMobject ooMobject = new OOMobject();
list.add(ooMobject);
try {
Thread.sleep(100);
} catch (Throwable ex) {
log.error("add oomobject wrongly", ex);
}
}
// log.info("add oomObject jvm memory...");
// printJvmMemoryInfo();
}
public static class OOMobject {
@SuppressWarnings("unused")
private byte[] bytes = new byte[1024 * 1024];
}
private static void printJvmMemoryInfo() {
long vmFree = 0;
long vmUse = 0;
long vmTotal = 0;
long vmMax = 0;
int byteToMb = 1024 * 1024;
Runtime rt = Runtime.getRuntime();
vmTotal = rt.totalMemory() / byteToMb;
vmFree = rt.freeMemory() / byteToMb;
vmMax = rt.maxMemory() / byteToMb;
vmUse = vmTotal - vmFree;
log.info("JVM内存情况:");
log.info("Jvm 内存已用的空间为:" + vmUse + " MB");
log.info("Jvm 内存空闲的空间为:" + vmFree + " MB");
log.info("Jvm 总内存空间为:" + vmTotal + " MB");
log.info("Jvm 总内存最大堆空间为:" + vmMax + " MB");
}
}
2)启动 JVM参数
-verbose:gc -Xlog:gc*=trace:file=gc.log:time,level,tags:filecount=50,filesize=100M -XX:NativeMemoryTracking=detail -XX:+HeapDumpOnOutOfMemoryError -Xmx512m -Xms128m
3) 分析
(1)打开分析工具显示的首页信息
展示大对象为 CopyOnWriteArrayList 类的实例,它若被释放的话,可以释放 Retained Size 的空间大小
(2)打开 Leak Suspects 窗口进行分析 (点击首页下部分的 Leak Suspects 链接)
最终找到 这个 0xe37ef740 数组中存储的是 OOMobject 对象,有问题的内是 JVMDemo 这个类,其使用了 CopyOnWriteArrayList 存储了 OOMObject 这个对象,因此可以看这块逻辑是否存在问题。
在 Leak Suspects 报告中还有更加直接的查看大对象,看下图:
显然大对象是 CopyOnWriteArrayList ,其中存储了 OOMObject 对象,现在需要分析是哪个类产生了这个大对象,按照图中选择 with incoming references 查看是哪些对象引用了这个对象。或者直接使用 “Path to GC Roots” 或 “Merge Shorts Paths to GC Roots” 查看直接到 GC Root 的链情况。
看到有3个线程用到这个对象,打开线程查看窗口,见下图:
找到那三个线程,最后发现是 0xe37c3790 这个线程中会产生 OOMObject 对象,并报出了 OOM 错误。
(3)打开 dominator tree 窗口进行分析 (点击首页下部分的 Dominator Tree 链接)从实例角度分析引用关系,可以对 retained heap 排序直接找大对象
找到之后,也是有右键选择 with incoming references 找使用这个大对象的 对象。分析方式和上节相同。
(4)打开 Histogram 窗口进行分析 (点击首页下部分的 Histogram,从类的维度查看类产生的所有实例占有内存情况)
对 Retained Heap 进行排序,看有哪个组定义的对象占最多,从图中可以看出 自定义的 OOMObject 占用内存最大。
参考:
https://blog.csdn.net/a303549861/article/details/82887431
原文地址:https://www.cnblogs.com/sandyflower/p/14588642.html
- 章神的私房菜之数据预处理
- 如何调用finecms指定栏目的描述关键词
- finecms栏目文章页seo设置
- Logistic回归实战篇之预测病马死亡率(一)
- DedeCMS后台500错误一种原因是不支持PHP5.3、5.4及以上版本
- finecms指定从第几篇文章开始调用5条记录,并调用文章所在栏目
- finecms如何调用多个栏目的子栏目
- 从傅立叶变换到Gabor滤波器
- 三个小时学会wordpress模板制作
- The each() function is deprecated报错的解决方法
- 书接上文:薛定谔的猫是如何诞生的?
- docker源码分析(3)---镜像(1)
- k8s源码分析-----kubelet(8)pod管理
- 大会 | DiracNets:无需跳层连接的ResNet
- java教程
- Java快速入门
- Java 开发环境配置
- Java基本语法
- Java 对象和类
- Java 基本数据类型
- Java 变量类型
- Java 修饰符
- Java 运算符
- Java 循环结构
- Java 分支结构
- Java Number类
- Java Character类
- Java String类
- Java StringBuffer和StringBuilder类
- Java 数组
- Java 日期时间
- Java 正则表达式
- Java 方法
- Java 流(Stream)、文件(File)和IO
- Java 异常处理
- Java 继承
- Java 重写(Override)与重载(Overload)
- Java 多态
- Java 抽象类
- Java 封装
- Java 接口
- Java 包(package)
- Java 数据结构
- Java 集合框架
- Java 泛型
- Java 序列化
- Java 网络编程
- Java 发送邮件
- Java 多线程编程
- Java Applet基础
- Java 文档注释
- Android自定义WaveProgressView实现水波纹加载需求
- CentOS8下的root密码快速修改方法
- Android开发之自定义刮刮卡实现代码
- Android ScrollView无法填充满屏幕的解决办法
- Android 监听屏幕是否锁屏的实例代码
- Android实现水波纹控件的方法
- Android中GridView布局实现整体居中方法示例
- Android SharedPreferences四种操作模式使用详解
- Ubuntu18.04下将 磁盘挂载在某目录下
- Android编程之绘图canvas基本用法示例
- Android 编译出错版本匹配问题解决办法
- Linux(CentOS7)使用 RPM 安装 mysql 8.0.11的教程
- Android Adapter里面嵌套ListView实例详解
- Centos7 安装达梦数据库的教程
- Android开发使用Handler实现图片轮播功能示例