java分析dump文件_干货分享丨jvm系列:dump文件深度分析
JVM dump
java内存dump是jvm运行时内存的一份快照,利用它可以分析是否存在内存浪费,可以检查内存管理是否合理,当发生OOM的时候,可以找出问题的原因。那么dump文件的内容是什么样的呢?我们一步一步来
获取JVM dump文件
获取dump文件的方式分为主动和被动
i.主动方式:
1.利用jmap,也是最常用的方式:jmap -dump:[live],format=b,file=
2.利用jcmd,jcmd GC.heap_dump
3.使用VisualVM,可以界面操作进行dump内存
4.通过JMX的方式
MBeanServerserver=ManagementFactory.getPlatformMBeanServer();
HotSpotDiagnosticMXBean mxBean=ManagementFactory.newPlatformMXBeanProxy(server, "com.sun.management:type=HotSpotDiagnostic", HotSpotDiagnosticMXBean.class);
mxBean.dumpHeap(filePath, live);
ii.被动方式:
被动方式就是我们通常的OOM事件了,通过设置参数-XX:+HeapDumpOnOutOfMemoryError -XX:HeapDumpPath=
dump文件分析
结构示意图
结构详解
dump文件是堆内存的映射,由文件头和一系列内容块组成
文件头
由musk, 版本,identifierSize, 时间4部分组成
1、musk:4个byte,内容为'J', 'A', 'V', 'A'即JAVA
2、version:若干byte,值有以下三种
" PROFILE 1.0\0",
" PROFILE 1.0.1\0",
" PROFILE 1.0.2\0"
3、identifierSize:4个byte数字,值为4或者8,表示一个引用所占用的byte数
4、time:8个byte,dump文件生成时间
说明:java一个类的成员变量有两种类型
基本类型(8种基本类型),它们占用byte数固定不变,每生成一个对象它们就需要给它们赋初始值,分配空间
是引用类型,表示一个对象,在类中只有一个引用,引用只是一个数值,所占用的空间大小为identifierSize,被引用对象即将在堆中的另一个地方
例如定义一个类
public class Person {
private int age;//4个byte
private String name;//identifierSize个byte
private double weight;//8个byte
}
当我们在new Person()的时候
它就需要申请一个空间,空间大小为 对象头大小+4+identifierSize+8个byte
对象大小的测量:
jdk提供一个测试对象占用内存大小的工具Instrumentation,但是Instrumentation没法直接引用到,需要通过agent来引用到
定义一个Premain类, javac Premain.java
//Premain.java
public class Premain {
public static java.lang.instrument.Instrumentation inst;
public static void premain(String args, java.lang.instrument.Instrumentation inst) {
Premain.inst= inst;
}
}
编写一个Manifest文件
manifest.mf
Manifest-Version: 1.0
Premain-Class: Premain
Can-Redefine-Classes: true
Can-Retransform-Classes: true
打包
jar -cmf manifest.mf premain.jar Premain.class
定义一个执行类, javac PersonTest.java
//PersonTest.java
public class PersonTest {
public static void main(String[] args) throws Exception {
Class clazz=Class.forName("Premain");
if (clazz != null) {
Person p=newPerson();
java.lang.instrument.Instrumentation inst= (java.lang.instrument.Instrumentation)clazz.getDeclaredField("inst").get(null);
System.out.println("person size:[" + inst.getObjectSize(p) + "]B");
System.out.println("class size:[" + inst.getObjectSize(p.getClass()) + "]B");
}
}
}
带agent执行
java -javaagent:premain.jar PersonTest
结果:
person size:[32]B
class size:[504]B
内容块
每个块都是块头和块体组成
块头
块头由1个byte的块类型,4个byte的时间time,4个byte的长度表示此内容块占用byte数
type类型一般有5种,字符串,类,栈桢,栈,及dump块
字符串,由identifierSize个byte的字符串id,后面是(length-identifierSize)个byte的字符串内容(后续对字符串是直接引用的这里面的id)
类,由4个byte的类序列(在栈桢中使用),identifierSize个byte的类id(解析类的时候用到),4个byte的序列id(暂未使用),identifierSize个byte的类名id
栈桢,由identifierSize个byte的桢id,identifierSize个byte的方法名id,identifierSize个byte的方法标识id,identifierSize个byte的类文件名id,4个byte的类序列,4个byte的行号
栈,由4个byte的栈序号,4个byte的线程序号,4个byte的桢数量,后面就是若干个identifierSize个byte的桢id
dump块就是所有对象的内容了,每个对象由1个byte的子类型,和对象内容结成,子类型有6种,gc root, 线程对象,类,对象,基本类型数组,对象数组
gc root
gc root有4种结构,8种类型
identifierSize个byte的对象id,类型有SYSTEM_CLASS,BUSY_MONITOR, 及未UNKNOWN
identifierSize个byte的对象id,4个byte的线程序列号,类型有NATIVE_STACK,THREAD_BLOCK
identifierSize个byte的对象id,4个byte的线程序列号,4个byte的栈桢深度,类型有JAVA_LOCAL,NATIVE_LOCAL
identifierSize个byte的对象id,identifierSize个byte的global refId(暂未使用),类型有NATIVE_STATIC
gc root示意图
gc root为垃圾收集追溯的源头,每个gc root都指向一个初始对象,无法追溯的对象是要被回收掉的
系统类,只有classLoader为null的类才是gc root,每个类都是一个gc root
线程栈,线程中方法参数,局部变量都是gc root,每个对象都是一个gc root
系统保留对象,每个对象都是一个gc root
类对象
1、基本信息:
identifierSize个byte的类对象id
4个byte的栈序列号,
identifierSize个byte的父类对象id,
identifierSize个byte的classLoader对象id,
identifierSize个byte的Signer对象id,
identifierSize个byte的protection domain对象id,
identifierSize个byte的保留id1和id2,
4个byte的类实例对象大小,
2个byte的常量个数,后面是每个常量的,2个byte的下标,1个byte的常量类型,和若干个byte的内容,内容根据类型来决定(boolean/byte为1个byte, char/short为2个byte,float/int为4个byte, double/long为8个byte,引用类型为identifierSize个byte)
10. 2个byte的静态变量个数,后面是每个静态变量的,identifierSize个byte的变量名id, 1个byte的变量类型,和若干个byte的内容,内容根据类型来决定(见类对象基本信息的第9条)
11. 2个byte的成员变量个数,后面是每个成员变量的,identifierSize个byte的变量名id,1个byte的变量类型
2、说明:
(1)类里面的常量很多地方都没有用上,所以常量个数一般为0
(2)类的静态变量的名称类型及值是放在类对象里面的,成员变量的名称和类型也是放在类对象里面的,但是实例的值是放在实例对象里面的
实例对象
1、基本信息:
identifierSize个byte的实例对象id
4个byte的栈序列号
identifierSize个byte的类id
4个byte的占用字节数
实例的变量的值
2、说明:
实例的值为实例对象的成员变量值,顺序为当前类的变量值,顺序为类对象基本信息中第11条中的顺序,然后是父类的变量值
变量的值基本类型都有默认值,引用类型默认值为0,占用字节数(见类对象基本信息的第9条)
基本类型数组
1、基本信息:
identifierSize个byte的数组对象id
4个byte的栈序列号
4个byte的数组长度
1个byte的元素类型
元素的值列表
2、说明:
元素的值(见类对象基本信息的第9条)
对象数组
1、基本信息:
identifierSize个byte的数组对象id
4个byte的栈序列号
4个byte的数组长度
identifierSize个byte的元素类id
元素的值列表
内存分配
当一个线程启动的时候,进程会去系统内存生成一个线程栈
每当发生一次方法调用,就会向栈中压入一个栈桢,当方法调用完之后,栈桢会退出
在运行过程中,如果有对象的new操作的时候,进程会去堆区申请一块内存
关于运行时内存的详细情况,可以查找相关的资料
内存回收规则
如果一个对象不能骑过gc root引用可达,那么这个对象就可能要被回收
对象回收规则包括
实例属性被实例引用,只有当实例被回收了实例属性才能被回收(只针对强引用)
类对象被实例引用,只有当一个类的所有实例都被回收了,类才能被回收
类对象的父类,classLoader对象,signer对象, protection domain对象被类引用,只有当类被回收了,这些才能被回收
局部变量(线程栈中)的作用域为一个大括号
public void test(){
Object a=newObject();//obj 1
Object b=newObject();//obj 2
{
Object c=newObject();//obj 3
a=null;//obj 1可以被回收了
}//obj 3可以回收了
}//obj 2可以被回收了
分析工具简介
分析dump文件,我们可以用jdk里面提供的jhat工具,执行
jhat xxx.dump
jhat加载解析xxx.dump文件,并开启一个简易的web服务,默认端口为7000,可以通过浏览器查看内存中的一些统计信息
一般使用方法
会列出一些功能,包括package下面各个类的概览,及各个功能导航
2、点击页面的堆内存统计
有一个表格,对象类型,实例个数,实例所占用内存大小,哪种类型的对象占用了内存最多一目了然
3、点击其中认为内存消耗太多的类名查看类详情
主要展现该类下面各个实例的大小,以及一些链接导航
4、点击references summary by type
如果某种类型的对象太多,那么有可能是引用它的那个类的对象太多
基本上一些简单页面的查询,结合原代码,就可以初步定位内存泄漏的地方
综上,dump文件结构还是比较简单的,这对于分析线程的执行情况非常有用,也是每一个Java程序员必须掌握的高级技能之一,你学会了吗?
【责任编辑:庞桂玉 TEL:(010)68476606】
点赞 0
java分析dump文件_干货分享丨jvm系列:dump文件深度分析相关推荐
- jmap 文件解析_干货分享丨jvm系列:dump文件深度分析
摘要:java内存dump是jvm运行时内存的一份快照,利用它可以分析是否存在内存浪费,可以检查内存管理是否合理,当发生OOM的时候,可以找出问题的原因.那么dump文件的内容是什么样的呢? JVM ...
- 【华为云技术分享】干货分享丨jvm系列:dump文件深度分析
摘要:java内存dump是jvm运行时内存的一份快照,利用它可以分析是否存在内存浪费,可以检查内存管理是否合理,当发生OOM的时候,可以找出问题的原因.那么dump文件的内容是什么样的呢? JVM ...
- android分享文件分享结果_干货分享丨DM删除归档日志文件的多种方法
上一期我们讲解了DM数据库归档日志的切换,本期我们讲解DM数据库归档日志文件的删除. 本章内容已在如下环境上测试: ①操作系统:中标麒麟7: ②数据库版本:达梦8: 相关关键字:DM数据库.归档日志文 ...
- dts数据库迁移工具_干货分享丨DM8 DTS工具使用小技巧
DTS工具的介绍 DM数据库为迁移提供了图形化工具--DTS.DTS可以从主流大型数据库迁移到DM.DM到DM.文件迁移到DM以及DM迁移到文件的功能,极大的简化了迁移操作,让数据迁移变得简单. DT ...
- 达梦数据库删除用户_干货分享丨DM8用户管理
原标题:干货分享丨DM8用户管理 用户介绍 安装创建达梦企业版数据库后,系统会默认创建四类数据库账号,分别是: SYS:达梦数据库内置管理用户,不能登录,数据库使用的大部分的数据字典和动态性能视图. ...
- java按需读取word文件_干货分享:ASP.NET CORE(C#)与Spring Boot MVC(JAVA)异曲同工的编程方式总结...
我(梦在旅途,http://zuowj.cnblogs.com; http://www.zuowenjun.cn)最近发表的一篇文章<.NET CORE与Spring Boot编写控制台程序应有 ...
- 关键词热度分析工具_干货分享丨关键词热度分析工具
不论我们是做竞价还是做自然优化,都需要对关键词的热度进行一番细致的分析.选择了不合适的关键词,可能就会浪费大量的时间和金钱,关键词的重要性不言而喻. 好啦,大家肯定会悄悄吐槽:谁不知道选择合适的关键词 ...
- win10任务栏透明_干货分享丨让你的win10桌面既整洁又美观
Hi,这里是一杯柔情. 从本月开始,每个月的第二周,我会分享一些实用干货. 这次给大家分享的是[怎么让你的win10电脑桌面变得既整洁又美观] 先来看一看效果图 应用不同的模板和壁纸,会有不同效果. ...
- linux mysql 误删系统文件恢复_干货分享丨如何恢复Linux下误删/etc目录数据
对于运维工作者来说,可能最让人担心的,是服务器宕机: 最让人无助的,是被DDOS: 而最让人心惊肉跳的,是rm -rf *这个命令-- 当你执行rm -rf命令时,万一哪个变量没赋值 听说过被删空服务 ...
- 杂项设备驱动框架_干货分享丨轻松玩转 Huawei LiteOS 传感框架
摘要:LiteOS传感框架将物联网终端设备上不同类型的传感器统一管理,通过抽象不同类型传感器接口,屏蔽其硬件细节,做到"硬件"无关性,非常方便于物联网设备的开发.维护和功能扩展. ...
最新文章
- Windows 日志高级筛选实践
- 中兴zxr10路由器重启命令_中兴交换机常用命令
- 详解java中Thread类,线程和进程的基本区别,多线程的好处,线程的五个生命周期,主线程和IDEA创建的Monitor Ctrl-Break守护线程;优雅地终止线程。死锁的产生
- mysql修改主键为自增长碰到的错误
- 机器学习第三篇:详解朴素贝叶斯算法
- java捕捉了异常_java 异常捕获与异常处理
- LeetCode 502. IPO(优先队列)
- struts框架学习过程中的问题。
- Servlet文件下载模板
- mysql 触发器检测表数据添加,进而调用存储过程检测数据,进而调用存储过程添加数据...
- 安装Nginx到linux服务器(Ubuntu)详解
- UML类图关系全面剖析
- 量子纠缠2——CHSH不等式
- 墨卡托投影、高斯-克吕格投影、UTM投影及我国分带方法
- photoshop设计精讲精练 学习笔记(二)
- Unity2018新功能之2D Animation2D动画
- DeepFool运行代码中间问题
- C# TSC TE244 PrintDocument 固定资产哑银不干胶标签打印
- Phobos Runtime Library
- 收集整理网络协议类型