jvm中jmap+MAT内存溢出实战(四)

一、内存溢出

1.堆内存溢出

堆内存中保存着对象,所以只要不停的往集合中存放对象,同时让gc不去回收对象,堆内存就会被撑爆。为了避免被gc回收,所以在一个方法中创建所以堆对象。

package com.jvm;

import java.util.ArrayList;
import java.util.List;

public class Test1 {
	
	public static void main(String[] args) {
		List<User> users=new ArrayList<User>();
		while(true) {
			users.add(new User("aa",12));
		}
	}
}

class User{
	public String name;
	public int age;
	
	public User(String name,int age) {
		this.name=name;
		this.age=age;
	}
}

为了缩短看见堆内存溢出异常的时间,启动时需要设置一下堆内存大小,如果使用的是eclipse,可以右键Run As-->Run configrations

点击argument;在vm argument窗口中写入: -Xms12m -Xmx12m

设置初始化堆大小12M,堆最大内存12M.

结果:

jvm中jmap+MAT内存溢出实战(四)

2.非堆内存溢出

非堆内存主要存在着Method,class的等文件信息。所以我们需要动态创建.class文件。

 

二、导出内存映像文件

1.参数

-XX:+HeapDumpOnOutOfMemoryError

-XX:HeapDumpPath=./

./:是路径,可以写绝对路径。

在添加jvm参数的窗口vm argument中加入上面两个参数。两个参数之间用空格隔开。

2.运行结果

jvm中jmap+MAT内存溢出实战(四)

3.在参数中写的路径下查找

会发现生成了一个以.hprof结尾的文件,这就是内存映像文件。

三、文件分析

1.安装MAT

可以在eclipse安装MAT的插件,也可以自行下载,为一个新的进程启动。

2.倒入.hprof

用MAT将我们生产的.hprof文件打开。

打开之后如图所示:

jvm中jmap+MAT内存溢出实战(四)

 

3.我们现在处于的overview窗口。

下面有三个:Action Reports Step By Step

我们点击报告Reports下面的Leak Suspects(怀疑溢出)

 

jvm中jmap+MAT内存溢出实战(四)

有a,b两块,Problem Suspect 1 (怀疑a块有问题)

jvm中jmap+MAT内存溢出实战(四)

占用了95%的空间。点开Details

jvm中jmap+MAT内存溢出实战(四)

列出了a模块的所以对象:com.jvm.User

从图可以看出内存溢出的主要原因是com.jvm.User类过多导致的。

四、jmap
1.上面设置jvm参数只能在堆内存溢出的时候才导出内存映像文件,但是很多时候我们需要在内存溢出之前(程序运行过程中)导出

jps  查看进程pid

jvm中jmap+MAT内存溢出实战(四)

jmap -dump:format=b,file=heap.hprof 23530

 

jvm中jmap+MAT内存溢出实战(四)

会在当前目录下生成一个heap.hprof的文件

使用jamp,可以在程序运行过程中随时到处内存映像文件

五、总结

堆内存溢出,一般情况下是对象过多,而对象又root可达,无法被GC回收,导致堆内存溢出。

上一篇:JVM性能调优监控工具jps、jstack、jmap、jhat、jstat、hprof使用详解


下一篇:java-执行jmap -heap选项时JVM是否停止?