Eclipse 的 MAT(Memory Analyzer Tool) 是一个内存分析工具,它可以分析内存中的对象、以及对象在内存的占用大小等等丰富功能 。
1 加载堆存储文件
安装好 MAT 之后,双击打开进入主界面,接着点击 File → Open Heap Dump 打开 JVM 的堆存储文件。打开之后,就会让我们选择希望首次进入时的报告页,这里使用默认:
2 概览(Overview)
点击 “Overview” ,打开概览页签:
概览页签的 Details 内容区,可以看到堆的大小、类、对象以及类加载器的个数。
接下来是个饼图,直观地描述了 dump 中占用内存最大的对象的前几名分布情况。
还可以使用鼠标移动并点击饼图中的某一块,进行更多分析操作:
3 类实例数列表(Histogram)
这里展示了每个类所对应的实例数列表。这里的每一列都可以点击,修改其排序方式。
第一列是类名,第二列是这个类所创建的实例数。
这里重点说说 Shallow Heap 与 Retained Heap。
Shallow Heap 指的是当前对象所占用的内存大小。一般来说,每种数据类型都有一个标准,比如 Integer 占用 4 字节;Long 占用 8 字节。
要说 Retained Heap,就必须先说说 Retained set。Retained set 是对象 X 以及与对象 X 有关的一系列对象集合,当对象 X 被 GC 回收后,这个集合中的对象,也会被一并回收。 Retained Heap 就是当对象 X 被 GC 回收后,被释放的内存大小。
举个例子,假设 A 与 B 对象是垃圾回收的根,从 C 到 H 对象,它们之间的关系是这样的:
那么 Retained set 中就会存在以下关系:
对象 X | 所对应的 Retained set |
---|---|
E | E、G |
C | C、D、E、F、G、H |
A,B | A、B 、C、D、E、F、G、H |
Histogram 默认会使用估算模式来计算 Retained Heap,因为只依赖于当前已检测到的集合中的对象数,所以会比精确计算快很多。
也可以点击 “计算器”图标,进行精确计算,不过很耗时,需要耐心等待哦O(∩_∩)O~
还可以使用正则表达式,过滤出我们想要看的 Class。比如,尝试在 ClassName 列的第一行输入 dbcp:
Histogram 可以按照不同的视角来展示该列表,默认是以类的形式,也可以按“超类”、按”类加载器”以及按“包”的形式,比如下例就是按“包”的形式来查找我们想要搜寻的类:
4 支配树(Dominator Tree)
Dominator Tree 中,显示的是 dump 中占用 Retained Heap 最多的大对象。
Dominator Tree 体现了对象实例之间的支配关系 。 在对象引用图中,如果所有指向对象 B 的路径都必须经过对象 A ,那么就认为对象 A 支配着对象 B。 如果对象 A 是离对象 B 最近的支配对象,那么对象 A 就是对象 B 的直接支配者 。 如果对象 A 支配着对象 B ,那么对象 A 的直接支配者也支配着对象 B。
左图表示对象之间的引用关系,右图表示左图所对应的支配树关系 。 对象 A 和 B 由 GC Roots 直接支配。因为在到对象 C 的路径中,即可以经过 A ,也可以经过 B ,所以对象 C 的直接支配者也是 GC Roots 。因此支配树关系图中的第一层是直连 A、B、C。 对象 F 与对象 D 相互引用,因为到对象 F 的所有路径必然经过对象 D ,所以,对象 D 是对象 F 的直接支配者 。 而到对象 D 的所有路径中,必然经过对象 C ,即使是从对象 F 到对象 D 的引用,从根节点出发,也是经过对象 C 的,所以,对象 D 的直接支配者为对象 C。 同理,对象 E 支配对象 G。 到达对象 H 的路径,即可以通过对象 D ,也可以通过对象 E ,因此对象 D 和 E 都不能直接支配对象 H ,而经过对象 C 既可以到达 D 也可以到达 E ,因此对象 C 为对象 H 的直接支配者 。
Dominator Tree 就是通过以上的逻辑判断,把对象之间的引用关系转换为对应的支配树关系的。