记一次JVM内存占用过高的优化经验

问题描述

1、生产环境一个WEB系统页面操作非常缓慢,遭到客户强烈投诉。经查服务器的内存占用达到80%-90%,初步怀疑是空闲内存空间不足导致。
2、经查内存占用排行榜前几位都是Java进程,其中有一个Java进程内存占用3.8GB左右。
3、用jmap导出内存后在MemoryAnalyzer中打开却显示只有500MB。

思考问题:为什么会相关那么大,如果要优化内存占用有什么办法或思路吗?

生产环境内存占用截图(第一个)

记一次JVM内存占用过高的优化经验_第1张图片

jmap导出结果在MemoryAnalyzer中打开(total只有500MB左右)

记一次JVM内存占用过高的优化经验_第2张图片

在MemoryAnalyzer中展开大对象分析如下:

记一次JVM内存占用过高的优化经验_第3张图片

解决思路及过程

对比进程内存和dump结果中的内存,我猜测是Java进程占用的内存涨上去之后就再也没有降下来。即使实际使用到的内存没有那么多,进程也一直占用那么多。

用jconsole分析内存情况如下:

记一次JVM内存占用过高的优化经验_第4张图片

用virtualvm分析内存情况如下:

记一次JVM内存占用过高的优化经验_第5张图片

从上面两张图的结果来看,实际使用的内存只占了进程内存的30%左右。证实前面的猜测正确。经过一翻搜索,找到内存配置参数,将它加入到java 命令后面即可。
-XX:NewRatio=2 -XX:MinHeapFreeRatio=8 -XX:MaxHeapFreeRatio=20

参数解析

-XX:NewRatio 设置年轻代和老年代的比值。默认值-XX:NewRatio=2,表示年轻代与老年代比值为1:2,年轻代占整个堆大小的1/3。我这里设置的就是默认值。
-XX:MaxHeapFreeRatio 空闲内存达到多少比值时,开始收缩JVM进程占用的内存空间。
-XX:MinHeapFreeRatio 空闲内存最少保留的比例值。我这里设置的是8%。此参数意思是无论如何进程内存会保留一定比例的空闲内存,进程内存收缩过程中,剩余内存达到这个阈值后,就会停止收缩。

优化后验证

(1)优化后,进程内存保持在1GB左右

记一次JVM内存占用过高的优化经验_第6张图片

(2)优化后的jconsole堆内存显示一直保持在300MB左右

记一次JVM内存占用过高的优化经验_第7张图片

(3)优化后的virtualvm内存

记一次JVM内存占用过高的优化经验_第8张图片

总结

虽然从事Java工作多年,Java进程占用内存多的问题也不是第一次遇见,但对这块知识一直没有深入研究。这是我第一次做这样的Java进程内存优化。
网友说JVM默认情况下内存空闲达到70%才会收缩。比方说java进程占用了1000MB,其中只用到了200MB,此时空闲内存有800MB超过了70%,它就会收缩。
如果用到了350MB,有650M空闲没有达到70%它就不收缩。

我把它改成了20%。
堆收缩是进程内存范围的收缩,GC是对象不用了把它的内存回收,GC和收缩是不一样的。
你调整初始大小,比如1GB,使用过程中(没有收缩的情况下)可能波动范围是500MB-3GB。
假设推扩大到3GB之后,实际使用内存回落到500MB,堆内存仍然占用3GB。如果按超过20空闲收缩的话它会变成500 + 500*0.2 = 600MB。

你可能感兴趣的:(Java,性能调优,内存优化,jvm,java,开发语言)