这个MAT工具的使用方式本文并没有给出,主要以内存排查过程以及案例为主,如果有需要MAT内存分析工具使用方式的话,可以另出一篇文章
我们JVM设置的最大堆内存是800M,虚拟机内存是3G,这里看到内存已经爆了,随之带来的则是频繁FULL GC导致的CPU 100%
可以看到老年代的使用率已经100%了
这里连续的几次查看,可以看到FULL GC的次数在疯狂的增长,而且FULL GC的平均时间也在增长,这就是CPU 100%的原因,因为FULL GC会导致stop-the-world的发生
查看占用最大的类
查看被谁引用了, 因为这个工具默认加载的就是不可被回收的对象,所以可以明显的看到具体的问题出在哪了
还可以换一种方式,就是如果是外部请求导致的,可以直接查看线程信息
使用jmeter工具并发请求 结果见上面排查结果
private static ThreadLocal threadLocal=new ThreadLocal();
/**
* ThreadLocal内存溢出 测试
* @return
*/
@GetMapping("/fullGc")
public String fullGc(){
for (int a = 0; a < 5; a++) {
//一次性产生5个8m 对象
threadLocal.set(new FullGcTest(new Byte[1024*1024*8]));
}
return "Success";
}
Druid在打开SQL监控后,会将执行的SQL语句保存下来,在一个LinkHashMap里面,以SQL语句为Key长期持有,默认情况下会保存1000条,这部分对象是不会被回收掉的,这里模拟极端情况,也就是这里每条SQL语句都非常大,可以通过控制随机数的大小来控制SQL语句的大小(根据JVM堆内存动态设置)
使用jmeter工具并发请求
/**
* druid测试 需要加入Druid连接池 并打开SQL监控
* @return
*/
@GetMapping("/druidTest")
public String druidTest(){
List ids=new ArrayList<>();
int i1 = new Random().nextInt(20000) + 500000;
for(int i=0;i
使用并发测试后就会发现内存溢出 (前提是设置的JVM内存大小适合,如果JVM内存非常大,这里SQL语句可能会撑不爆,便不会发生内存溢出,但此时也可以导出dump查看不可回收对象)
可以看到全是执行过的SQL
本文的两个案例都是可能会导致内存溢出,因为都会产生不可回收对象,虽然DRUID连接池这个例子有点极端,但也未尝不是一个会产生不可回收对象的例子,而且使用还极为广泛,内存溢出的主要元凶还是产生了大量的不可回收对象,避免产生不可回收对象才是我们平时开发中需要注意的。
如果项目内没有多少不可回收对象,要如何优化内存才能达到最高的吞吐量呢?也就是所谓的JVM调优,我们下篇继续