在平时开发、测试过程中、甚至是生产环境中,有时会遇到OutOfMemoryError,Java堆溢出了,这表明程序有严重的问题。我们需要找造成OutOfMemoryError原因。一般有两种情况:
1、内存泄露,对象已经死了,无法通过垃圾收集器进行自动回收,通过找出泄露的代码位置和原因,才好确定解决方案;
2、内存溢出,内存中的对象都还必须存活着,这说明Java堆分配空间不足,检查堆设置大小(-Xmx与-Xms),检查代码是否存在对象生命周期太长、持有状态时间过长的情况。
以上是处理Java堆问题的思路,具体是怎么进行分析,这里介绍的是使用Eclipse Memory Analyzer tool(MAT)工具分析的过程。
2、内存溢出,内存中的对象都还必须存活着,这说明Java堆分配空间不足,检查堆设置大小(-Xmx与-Xms),检查代码是否存在对象生命周期太长、持有状态时间过长的情况。
以上是处理Java堆问题的思路,具体是怎么进行分析,这里介绍的是使用Eclipse Memory Analyzer tool(MAT)工具分析的过程。
1.生成dump文件
通过jvm参数--XX:-HeapDumpOnOutOfMemoryError可以让JVM在出现内存溢出是Dump出当前的内存转储快照;
用jmap生产dump文件,win通过任务管理器查看tomcat的进程pid,linux用ps命令查看进程pid,然后用jmap命令
通过jvm参数--XX:-HeapDumpOnOutOfMemoryError可以让JVM在出现内存溢出是Dump出当前的内存转储快照;
用jmap生产dump文件,win通过任务管理器查看tomcat的进程pid,linux用ps命令查看进程pid,然后用jmap命令
(Java5:jmap -heap:format=b <pid>;Java6:jmap -dump:format=b,file=HeapDump.bin <pid>)。
生成之后结果如下:
生成之后结果如下:
这说明在这个路径下面生成了一个dump文件
2.开始使用MAT进行OOM分析
第一步,启动mat ,选择File->Open Heap Dump 选择你的dump文件。下面开始等待,mat解析dump文件需要花一些时间,在解析的同时会在硬盘上写入一些解析结果文件,这样下次打开时速度会快很多。有时候mat在解析过程中可能会出现出错的情况,这个时候可以将那些临时文件删除以后重试第一步,如果你的rp够好的话重试也许会解析成功。
第二步,查看内存泄漏分析报表。mat解析完成以后会出现如下图的提示:
因为我们就是为了查找内存泄漏的问题,所以保持默认选项直接点“Finish”就可以。
第二步,查看内存泄漏分析报表。mat解析完成以后会出现如下图的提示:
![](http://dl.iteye.com/upload/picture/pic/80860/8c91333f-82a6-32af-9481-def0a9db0d1c.png)
因为我们就是为了查找内存泄漏的问题,所以保持默认选项直接点“Finish”就可以。
3.分析图解
通过MAT打开dump出来的内存文件,打开后如下图:
![点击查看原始大小图片](http://dl.iteye.com/upload/attachment/0066/6787/84bb3118-17b2-366c-923f-8fc3fb03705b.png)
从上图可以看到它的大部分功能。
1. Histogram可以列出内存中的对象,对象的个数以及大小。
2. Dominator Tree可以列出那个线程,以及线程下面的那些对象占用的空间。
3.Top consumers通过图形列出最大的object。
4.Leak Suspects通过MA自动分析泄漏的原因。
Histogram如下图:
Objects:类的对象的数量。
Shallow size:就是对象本身占用内存的大小,不包含对其他对象的引用,也就是对象头加成员变量(不是成员变量的值)的总和。
Retained size:是该对象自己的shallow size,加上从该对象能直接或间接访问到对象的shallow size之和。换句话说,retained size是该对象被GC之后所能回收到内存的总和。
我们发现ThreadLocal和bingo.persister.dao.Daos类的对象占用了很多空间。
![点击查看原始大小图片](http://dl.iteye.com/upload/attachment/0066/6785/970df6ba-575b-3fb9-9a0d-4cc34fcead6b.png)
Dominator Tree如下图:
我们发现quartz的定时器的工作线程(10个)占了很多的内存空间
Top consumers如下图:
这里显示了内存中最大的对象有哪些,他们对应的类是哪些,类加载器classloader是哪些。
有些时候,我们在这里就可以看到代码泄露的位置。
![点击查看原始大小图片](http://dl.iteye.com/upload/attachment/0066/6796/ef17e5cd-d7ec-3d8b-a07b-70c65fc7783e.png)
Leak Suspects如下图:
从那个饼图,该图深色区域被怀疑有内存泄漏,可以发现整个heap才250M内存,深色区域就占了34%。后面的描述,告诉我们quartz线程占用了大量内存,并指出system class loader加载的"java.lang.ThreadLocal"实例的内存中聚集(消耗空间),并建议用关键字"java.lang.ThreadLocal$ThreadLocalMap$Entry[]"进行检查。所以,MAT通过简单的报告就说明了问题所在。
![点击查看原始大小图片](http://dl.iteye.com/upload/attachment/0066/6791/6be4bf4e-d7f6-33f6-a5b1-2aef78d7ab84.png)
通过Leak Suspects的Problem Suspect 1点击【Details »】,
如下图如下图所示的上下文菜单中选择 List objects -> with outgoning references, 查看ThreadLocal都应用了些什么对象。
![点击查看原始大小图片](http://dl.iteye.com/upload/attachment/0066/6794/77779fa4-20d9-329e-b7fb-d1117d494592.png)
现在看到ThreadLocal中引用的对象如下图:
是dao对象
ps:该dao对象包含一个轻量级的ORM关系内容,所以Retained size比较大。
![点击查看原始大小图片](http://dl.iteye.com/upload/attachment/0066/6793/9cf2d100-bf3f-394c-b86f-5577707d7778.png)
下面继续查看dao的gc ROOT
如下图所示的上下文菜单中选择 Path To GC Roots -> exclude weak references, 过滤掉弱引用,因为在这里弱引用不是引起问题的关键。
![点击查看原始大小图片](http://dl.iteye.com/upload/attachment/0066/6803/f0a604d4-506f-37e9-a88e-333c730be019.png)
从下图中,可以看到在org.quartz.simpl.SimpleThreadPool中保存了daos的引用。所以可以得出是是因为定时器在运行的过程中持有大量的Daos对象应起了内存泄露。为什么会有那么多的Daos呢,Daos不是一个无状态的单例的、可以重用的吗?继续查看spring配置文件发现Daos的bean配置成scope="prototype",导致定时任务又是每次调用都生产新的Daos实例。由于是Daos是无状态的,修改为单例的,问题解决。
![点击查看原始大小图片](http://dl.iteye.com/upload/attachment/0066/6801/c9820b9e-1f42-30cc-9d32-c3c390288b3f.png)
相关推荐
### Eclipse Memory Analyzer (MAT) 分析 Tomcat 内存溢出详解 #### 一、引言 在软件开发过程中,特别是在 Java 应用程序中,内存管理是非常关键的一环。当应用程序遭遇 `OutOfMemoryError` 异常时,通常意味着...
Eclipse Memory Analyzer 是一个功能丰富且轻量的 Java 堆内存分析工具,可以用来辅助发现内存泄漏减 少内存占用。 使用 Memory Analyzer 来分析生产环境的 Java 堆转储文件,可以从数以百万计的对象中快速计算出对 ...
Eclipse Memory Analyzer,通常简称为MAT(Memory Analyzer Tool),是一款由Eclipse基金会开发的强大内存分析工具,主要用于诊断Java应用程序的内存泄漏和优化内存使用。它提供了深入的heap dump分析,帮助开发者...
点到 Y 的所有路径都必须经过 X。➢根对象(root objects...通过深入理解和熟练运用这些知识点,你可以有效地利用 Eclipse Memory Analyzer 工具,诊断和解决 Java 应用程序中的内存泄漏问题,提高系统的稳定性和性能。
Eclipse Memory Analyzer(MAT)是一款强大的Java内存分析工具,它被广泛用于诊断和解决Java应用程序中的内存泄漏问题。MAT 1.8.1是适用于macOS的特定版本,旨在帮助开发者深入理解并优化他们的Java应用内存使用情况...
Eclipse Memory Analyzer(内存分析器)是一款专门为Java堆内存分析而设计的工具,它可以协助开发者快速地分析内存泄漏问题,通过生成的报告指出潜在的内存泄漏可疑点。Memory Analyzer可以单独使用,也可以作为...
Eclipse Memory Analyzer(MAT)是一款专业且功能强大的Java堆转储(heap dump)分析工具,由Eclipse基金会开发。它旨在帮助Java开发者诊断和解决内存泄漏、内存占用过高等问题,提升应用程序的性能和稳定性。MAT...
《Eclipse Memory Analyzer》中文文档是一份详尽的指南,主要针对Java开发人员,特别是那些在处理内存管理和优化方面遇到挑战的开发者。这份文档共有93页,旨在帮助用户理解和利用MAT (Memory Analyzer Tool)有效地...
MAT(Memory Analyzer Tool)是Eclipse项目开发的一款强大的Java内存分析工具,主要用于诊断Java应用程序的内存泄漏和性能问题。在标题中提到的“Eclipse Memory Analyzer Version 1.7.0.rar”是一个压缩包,其中...
MAT是分析Java堆内存的一个工具,全称是 The Eclipse Memory Analyzer Tool,用来帮助分析内存泄漏和减少内存消耗。使用MAT分析Java堆快照,可以快速计算出对象的保留大小(Retained Sizes),查找到阻止对象被回收...
Eclipse Memory Analyzer(MAT,全称Memory Analyzer Tool)是一款强大的Java内存分析工具,尤其在Mac平台上,它提供了独立于Eclipse环境的版本,方便开发者直接使用。MAT的主要目标是帮助开发者诊断和解决Java应用...
Eclipse Memory Analyzer(MAT)是一款强大的Java内存分析工具,尤其适用于诊断和解决Java应用程序中的内存泄漏问题。这个工具是Eclipse基金会的一个项目,提供了一个直观的用户界面来帮助开发者深入理解堆内存的...
Eclipse Memory Analyzer(MAT)是一款强大的Java内存分析工具,它能帮助开发者诊断和解决与内存泄漏、内存占用过高以及垃圾收集性能问题相关的问题。在Java应用程序中,如果内存管理不当,可能会导致系统运行缓慢,...
Eclipse Memory Analyzer(MAT)是一款强大的Java虚拟机(JVM)内存分析工具,它能够帮助开发者定位并解决Java应用程序中的内存泄漏问题。MAT通过解析.hprof文件,即Java虚拟机生成的内存快照,来展示堆内存的详细...
MemoryAnalyzer,即MAT(Memory Analyzer Tool),是一款由Eclipse基金会开发的强大的Java内存分析工具,尤其适用于IBM JVM(openj9)上的heap dump文件分析。本文将详细介绍MemoryAnalyzer的特性和功能,以及如何...
MemoryAnalyzer,通常简称为MAT,是由Eclipse基金会开发的一款强大的Java堆内存分析工具。它能帮助开发者定位并解决Java应用程序中的内存泄露问题。MAT通过分析heap dump文件(即Java虚拟机的堆内存快照),提供详尽...
MemoryAnalyzer,简称MAT,是一款由Eclipse基金会开发的专业内存分析工具,其版本号为1.7.0.201706130745,主要用于Java应用程序的内存泄漏检测和性能优化。该软件的核心功能在于帮助开发者找出程序运行过程中可能...
总结来说,MemoryAnalyzer是一款强大的JVM堆内存分析工具,它的功能全面,易于使用,能有效地帮助开发者排查和解决内存相关的问题,提升Java应用的性能和稳定性。熟练掌握MAT的使用,是每个Java开发者必备的技能之一...
Eclipse Memory Analyzer(MAT)是Eclipse基金会提供的一款强大的Java内存分析工具,它主要用于诊断Java应用程序的内存泄漏问题。MAT通过分析heap dump文件,能够帮助开发者深入理解内存消耗的细节,找出内存占用过...