java 第三方内存分析工具MAT怎么用

发布时间:2021-10-20 16:48:59 作者:柒染
来源:亿速云 阅读:188

java 第三方内存分析工具MAT怎么用,相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。

tags: java,troubleshooting,monitor,mat


一句话概括:MAT是一个强大的内存分析工具,可以快捷、有效地帮助我们找到内存泄露,减少内存消耗分析工具,下文将进行讲解。

1 引言

之前的文章有提过,内存中堆的使用情况是应用性能监测的重点,而对于堆的快照,可以dump出来进一步分析,总的来说,一般我们对于堆dump快照有三种方式:

那么,对于dump出来的文件,如何对它们进行分析?jvisualvm可以直接装入快照文件进行分析,而本文所介绍的MAT,相对来说内存分析功能更强大,它可以自动检测有可能发生问题(特别是内存溢出、内存泄露)的地方,也可以详细查看类内存占用情况,方法级的调用情况等,是一个分析内存状况的不可多得的工具。

2 MAT工具介绍

MAT(Memory Analyzer tool)是一款内存分析器工具,它是一个快速且功能丰富的堆内存分析器,帮助我们查找内存泄漏和分析高内存消耗问题。官方网站是:https://www.eclipse.org/mat/,有兴趣可以上去看一下。使用MAT,可以轻松实现以下功能:

3 MAT工具安装

MAT安装有两种方式,一种是以eclipse插件方式安装,一种是独立安装。在MAT的官方文档中有相应的安装文件下载,下载地址为:https://www.eclipse.org/mat/downloads.php

4 MAT工具使用

MAT定位是内存分析工具,它的主要功能就是对内存快照进行分析,帮助我们找到有可能内存溢出或内存泄漏的地方,因此,找到占用内存大的对象和找出无法回收的对象是其主要目的。MAT官方文档,地址如下:https://help.eclipse.org/2019-06/index.jsp?topic=/org.eclipse.mat.ui.help/welcome.html,对MAT的使用进行描述,有兴趣的同学可以上去看看。下面主要对MAT的常用概念、常用的功能进行介绍。

下文中,以java-monitor-example(https://github.com/mianshenglee/my-example/tree/master/java-monitor-example)为例,此示例是一个简单的spring boot工程,里面的一个controller中的user/oom接口调用的service对象通过List成员不断地添加User对象,最终导致OOM的发生,应用的启动参数是-Xms64m -Xmx64m -XX:+HeapDumpOnOutOfMemoryError -XX:HeapDumpPath=${APP_HOME}/logs/heapdump.hprof

4.1 MAT相关概念说明

4.1.1 内存泄漏与内存溢出

4.1.2 引用(强引用,软引用,弱引用,虚引用)

4.1.3 shallow heapretained heap

4.1.4 outgoing referencesincoming references

4.1.5 Dominator Tree

将对象树转换成Dominator Tree能帮助我们快速的发现占用内存最大的块,也能帮我们分析对象之间的依赖关系。Dominator Tree有以下几个定义:

对象树和Dominator tree的对应关系如下:

java 第三方内存分析工具MAT怎么用

如上图,因为A和B都引用到C,所以A释放时,C内存不会被释放。所以这块内存不会被计算到A或者B的Retained Heap中,因此,对象树在转换成Dominator tree时,会A、B、C三个是平级的。

4.1.6 Garbage Collection Roots(GC root)

在执行GC时,是通过对象可达性来判断是否回收对象的,一个对象是否可达,也就是看这个对象的引用连是否和GC Root相连。一个GC root指的是可以从堆外部访问的对象,有以下原因可以使一个对象成为GC root对象。

4.2 Leak Suspects自动分析泄漏

当发生OOM获取到heapdump.hprof文件或者手动dump出文件后,使用MAT打开文件。打开后默认会提示是否进行内存泄漏检测报告(如果打开Dump时跳过了的话,也可以从其它入口进入工具栏上的 Run Expect System Test -> Leak Suspects),如下图所示:

java 第三方内存分析工具MAT怎么用

选择是后进入报告内容,此报告内容会帮我们分析的可能有内存泄露嫌疑的地方,它会把累积内存占用较大的通过饼状图显示出来。如下图所示:

java 第三方内存分析工具MAT怎么用

如上图,报告已指出UserService占用了76.73%的内存,而这些内存都在Object[]这个数组中。因此,很大一种可能是这个对象中的数组数量过大。点击Details可以查看更详细的内容:

java 第三方内存分析工具MAT怎么用

在此详细而,Shortest Paths To the Accumulation Point可以显示出到GC roots的最短路径,由此可以分析是由于和哪个GC root相连导致当前Retained Heap占用相当大的对象无法被回收,本示例中,GC root是线程的本地变量(java local)。Accumulated Objects in Dominator TreeDominator Tree为视图,可以方便的看出受当前对象“支配”的对象中哪个占用Retained Heap比较大。图中可看出对象都在ArrayList中,而ArrayList下有Object数组,数组下是User对象。此可以知道问题出在哪里了。需要针对这个位置,查看代码,找出导致这个数组存储的User过量的原因。

注:在原始堆转储文件的目录下,MAT已经将报告的内容压缩打包到一个zip文件,命名为“xxx_Leak_Suspects.zip”,整个报告是一个HTML格式的文件,可以用浏览器直接打开查看,可以方便进行报告分发、分享。

4.3 histogram视图查看对象个数与大小

点击Overview页面Actions区域内的“Histogram视图”或点击工具栏的“histogram”按钮,可以显示直方图列表,它以Class类的维度展示每个Class类的实例存在的个数、 占用的Shallow heapRetained内存大小,可以分别排序显示。如下图所示:

java 第三方内存分析工具MAT怎么用

Shallow HeapRetained Heap的概念前面已经讲过。 对于java的对象,其成员基本都是引用。真正的内存都在堆上,看起来是一堆原生的byte[], char[], int[],因此如果只看对象本身的内存,数量都很小,而多数情况下,在Histogram视图看到实例对象数量比较多的类都是一些基础类型(通常都排在前面),如char[]、String、byte[],所以仅从这些是无法判断出具体导致内存泄露的类或者方法的。从上图中,可以直接看到User对象的数量很多,有时不容易看出来的,可以使用工具栏中的group result by 可以以super class,class loader等排序,需要特别注意自定义的classLoader,如下图:

java 第三方内存分析工具MAT怎么用

通常,如果Histogram视图展示的数量多的实例对象不是基础类型,而是用户自定义的类或者有嫌疑的类,就要重点关注查看。想进一步分析的,可以右键,选择使用List objectsMerge Shortest Paths to GC roots等功能继续钻取数据。其中list objects分别有outgoing referencesincoming references,可以找出由这个对象出去的引用及通过哪些引用到这个对象的。Merge Shortest Paths to GC roots可以排除掉所有不是强引用的,找到这个对象的到GC root的引用路径。最终目的就是找到占用内存最大对象和无法回收的对象,计算从垃圾收集器根到相关对象的路径,从而根据这个对象路径去检查代码,找出问题所在。

4.4 Dominator Tree视图

点击Overview页面Actions区域内的“Dominator Tree视图”或点击工具栏的“open Dominator Tree”按钮,可以进入 Dominator Tree视图。该视图以实例对象的维度展示当前堆内存中Retained Heap占用最大的对象,以及依赖这些对象存活的对象的树状结构。如下图:

java 第三方内存分析工具MAT怎么用

视图展示了实例对象名、Shallow Heap大小、Retained Heap大小、以及当前对象的Retained Heap在整个堆中的占比。该图是树状结构的,当上一级的对象被回收,那么,它引用的子对象都会被回收,这也是Dominator的意义,当父节点的回收会导致子节点也被回收。通过此视图,可以很方便的找出占用Retained Heap内存最多的几个对象,并表示出某些objects的是因为哪些objects的原因而存活。本示例中,可以看出是由于UserService中的ArrayList引用的数组存储过量的User对象。

4.5 线程视图查看线程栈运行情况

点击工具栏的“齿轮”按钮,可以打开Thread Overview视图,可以查看线程的栈帧信息,包括线程对象/线程栈信息、线程名、Shallow HeapRetained Heap、类加载器、是否Daemon线程等信息。结合内存Dump分析,看到线程栈帧中的本地变量,在左下方的对象属性区域还能看到本地变量的属性值。如按上面的示例(shortest paths to GC root),知道是由于线程Thread-12GC-root占用内存大,在线程视图中,就可以着重看它的属性情况,如下图:

java 第三方内存分析工具MAT怎么用

由上图可以看到此线程调用WithOOM方法,使用了变量UserService,而变量使用了userList,它包含了大量的User对象,占用retained heap很大。

对于java应用的内存分析,需要对java应用的内存进行dump操作,生成内存快照后,使用MAT进行分析,找出大对象,找到内存泄漏或溢出的地方,从而分析代码,解决问题。本文对MAT的使用场景,基本概念,安装、使用进行了详细介绍,大家可以自己安装一下,写个小示例或者拿本文中的示例,实践一下。通过本文,希望可以帮助大家更方便,更有效率地分析内存,解决java应用的内存故障问题。

看完上述内容,你们掌握java 第三方内存分析工具MAT怎么用的方法了吗?如果还想学到更多技能或想了解更多相关内容,欢迎关注亿速云行业资讯频道,感谢各位的阅读!

推荐阅读:
  1. Java异常之OutOfMemoryError怎么解决
  2. Java 堆内存溢出原因分析

免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。

java mat

上一篇:如何用迭代器设计模式提升Python性能

下一篇:如何理解配置中心的技术选型

相关阅读

您好,登录后才能下订单哦!

密码登录
登录注册
其他方式登录
点击 登录注册 即表示同意《亿速云用户服务条款》