• 记一次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。

  • 相关阅读:
    SpringBoot对接阿里云OSS上传文件以及回调(有坑)
    计算机学院院长第一课——前辈的经验
    c++新经典—c++基本语言
    springboot+大学生就业规划系统 毕业设计-附源码191451
    c++征途 --- 函数提高
    力扣--有效的括号
    CSS flex使用详解
    如何在面试中胜出?接口自动化面试题安排上
    Pandas数据类型-DataFrame数据编辑
    如何修炼成“神医”——《OceanBase诊断系列》之一
  • 原文地址:https://blog.csdn.net/hansonjan/article/details/127767528