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

问题描述

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

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

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

在这里插入图片描述

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

在这里插入图片描述

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

在这里插入图片描述

解决思路及过程

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

用jconsole分析内存情况如下:

在这里插入图片描述

用virtualvm分析内存情况如下:

在这里插入图片描述

从上面两张图的结果来看,实际使用的内存只占了进程内存的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左右

在这里插入图片描述

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

在这里插入图片描述

(3)优化后的virtualvm内存

在这里插入图片描述

总结

虽然从事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。

  • 3
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 调优 JVM 的方法有很多,常用的方法如下: 1. 调整堆内存大小。可以使用 `-Xms` 和 `-Xmx` 参数来调整堆内存的初始大小和最大大小。如果堆内存过小,会导致频繁的 GC,从而导致 CPU 占用。 2. 调整 GC 策略。JVM 中有很多不同的 GC 算法,如 Serial GC、Parallel GC、CMS GC 等。每种算法都有自己的优缺点。可以使用 `-XX:+UseSerialGC` 或者 `-XX:+UseParallelGC` 等参数来选择不同的 GC 算法。 3. 调整 JVM 参数。JVM 还有很多其他的参数,比如 `-XX:+PrintGC` 可以在控制台输出 GC 的日志,`-XX:+PrintGCDetails` 可以输出详细的 GC 日志,方便我们分析问题。 4. 使用 profiler 分析 CPU 占用情况。有很多 profiler 工具可以帮助我们分析 CPU 占用情况。比如 Java Flight Recorder、JProfiler 等。使用这些工具可以帮助我们找到 CPU 占用的原因,从而调优 JVM。 ### 回答2: 要优化JVM来降低CPU占用的问题,可以从以下几个方面进行优化: 1. 调整内存设置:通过适当调整JVM内存设置来避免内存压力过大,导致频繁的垃圾回收。可以通过修改-Xms和-Xmx参数来调整JVM的初始堆和最大堆大小,需要根据实际情况进行调整。 2. 减少对象创建:过多的对象创建会导致频繁的垃圾回收,从而造成CPU占用。可以通过优化代码,避免不必要的对象创建,使用对象池等方式减少对象的产生。 3. 优化算法和数据结构:选择合适的算法和数据结构可以减少CPU的使用率。对于一些耗时的操作,可以考虑使用更效的算法来替代。 4. 多线程优化:合理的使用多线程可以充分利用多核CPU的性能,减少单个线程的CPU占用率。可以将一些独立、并行的任务放入不同的线程中进行处理。 5. 使用性能分析工具:可以使用性能分析工具来检测和分析CPU占用的原因,如JProfiler、VisualVM等。通过定位到具体的问题,可以有针对性地进行优化。 总之,JVM优化需要综合考虑内存设置、对象创建、算法数据结构、多线程等多个因素。通过适当的调整和优化,可以降低CPU占用的问题,提系统的性能。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值