• JVM实战:CMS和G1的物理内存归还机制


    前言

    公司有一个系统使用的是CMS垃圾回收器,JVM初始堆内存不等于最大堆内存,但通过监控信息发现:在经过一次FullGC之后,服务器物理内存剩余空间并未提升,按照我之前的理解FullGC之后JVM进程会释放的内存一部分还给物理内存,下面通过几个实验来对比验证一下CMS和G1的物理内存归还机制

    测试代码

    public class MemoryRecycleTest {
    
        static volatile List list = new ArrayList<>();
    
        public static void main(String[] args) {
            //指定要生产的对象大小为512M
            int count = 512;
    
            //新建一条线程,负责生产对象
            new Thread(() -> {
                try {
                    for (int i = 1; i <= 10; i++) {
                        System.out.println(String.format("第%s次生产%s大小的对象", i, count));
                        addObject(list, count);
                        //休眠40秒
                        Thread.sleep(i * 10000);
                    }
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }).start();
    
            //新建一条线程,负责清理List,回收JVM内存
            new Thread(() -> {
                for (; ; ) {
                    //当List内存到达512M,就通知GC回收堆
                    if (list.size() >= count) {
                        System.out.println("清理list.... 回收jvm内存....");
                        list.clear();
                        //通知GC回收
                        System.gc();
                        //打印堆内存信息
                        printJvmMemoryInfo();
                    }
                }
            }).start();
    
            //阻止程序退出
            try {
                Thread.currentThread().join();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    
        public static void addObject(List list, int count) {
            for (int i = 0; i < count; i++) {
                OOMobject ooMobject = new OOMobject();
                //向List添加一个1M的对象
                list.add(ooMobject);
                try {
                    //休眠100毫秒
                    Thread.sleep(100);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }
    
        public static class OOMobject {
            //生成1M的对象
            private byte[] bytes = new byte[1024 * 1024];
        }
    
        public static void printJvmMemoryInfo() {
            //虚拟机级内存情况查询
            long vmFree = 0;
            long vmUse = 0;
            long vmTotal = 0;
            long vmMax = 0;
            int byteToMb = 1024 * 1024;
            Runtime rt = Runtime.getRuntime();
            vmTotal = rt.totalMemory() / byteToMb;
            vmFree = rt.freeMemory() / byteToMb;
            vmMax = rt.maxMemory() / byteToMb;
            vmUse = vmTotal - vmFree;
            System.out.println("");
            System.out.println("JVM内存已用的空间为:" + vmUse + " MB");
            System.out.println("JVM内存的空闲空间为:" + vmFree + " MB");
            System.out.println("JVM总内存空间为:" + vmTotal + " MB");
            System.out.println("JVM总内存最大堆空间为:" + vmMax + " MB");
            System.out.println("");
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60
    • 61
    • 62
    • 63
    • 64
    • 65
    • 66
    • 67
    • 68
    • 69
    • 70
    • 71
    • 72
    • 73
    • 74
    • 75
    • 76
    • 77
    • 78
    • 79
    • 80
    • 81
    • 82
    • 83
    • 84

    JDK8 CMS

    JVM参数

    -Xms128M -Xmx2048M -XX:+UseConcMarkSweepGC
    
    • 1

    控制台打印的内容

    第1次生产512大小的对象
    清理list.... 回收jvm内存....
    
    JVM内存已用的空间为:6 MB
    JVM内存的空闲空间为:1202 MB
    JVM总内存空间为:1208 MB
    JVM总内存最大堆空间为:1979 MB
    
    第2次生产512大小的对象
    清理list.... 回收jvm内存....
    
    JVM内存已用的空间为:3 MB
    JVM内存的空闲空间为:1097 MB
    JVM总内存空间为:1100 MB
    JVM总内存最大堆空间为:1979 MB
    
    第3次生产512大小的对象
    清理list.... 回收jvm内存....
    
    JVM内存已用的空间为:3 MB
    JVM内存的空闲空间为:706 MB
    JVM总内存空间为:709 MB
    JVM总内存最大堆空间为:1979 MB
    
    第4次生产512大小的对象
    清理list.... 回收jvm内存....
    
    JVM内存已用的空间为:3 MB
    JVM内存的空闲空间为:120 MB
    JVM总内存空间为:123 MB
    JVM总内存最大堆空间为:1979 MB
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31

    VisualVM监控的堆内存情况

    在这里插入图片描述

    从图中堆内存的情况可以看出,在JDK8+CMS的配置下,JVM并不是立马归还内存给到操作系统,而是随着FullGC次数的增多逐渐归还,最终会全部归还

    JDK8 G1

    JVM参数

    -Xms128M -Xmx2048M -XX:+UseG1GC
    
    • 1

    VisualVM监控的堆内存情况

    在这里插入图片描述

    在JDK8+G1的配置下,JVM都是在每一次FullGC后全部归还物理内存

    JDK11 CMS

    JVM参数

    -Xms128M -Xmx2048M -XX:+UseConcMarkSweepGC
    
    • 1

    VisualVM监控的堆内存情况

    在这里插入图片描述

    在JDK11+CMS的配置下和JDK8+CMS的情况相同(JVM并不是立马归还内存给到操作系统,而是随着FullGC次数的增多逐渐归还,最终会全部归还)

    JDK11提供了一个JVM参数ShrinkHeapInSteps 。通过这个参数,可以在GC之后渐进式的归还内存给到操作系统。JDK11下,此参数默认开启。可以把此参数关闭,看下堆内存的变化情况:

    -Xms128M -Xmx2048M -XX:+UseConcMarkSweepGC -XX:-ShrinkHeapInSteps
    
    • 1

    VisualVM监控的堆内存情况

    在这里插入图片描述

    在JDK11+CMS的配置下,关闭ShrinkHeapInSteps参数后,JVM都是在每一次FullGC后全部归还物理内存

    JDK11 G1

    由于JDK11默认使用的是G1垃圾回收器,所以这里只设置了初始堆内存和最大堆内存

    JVM参数

    -Xms128M -Xmx2048M
    
    • 1

    VisualVM监控的堆内存情况

    在这里插入图片描述

    1)JDK11默认的ShrinkHeapInSteps是默认开启的,但这里看堆内存变化并不是渐进的缩小的。 所以在G1回收器下,ShrinkHeapInSteps是无效的。 如果我们手动关闭ShrinkHeapInSteps参数,发现堆内存变化和上面这个类似

    2)JDK11下的G1和JDK8下的G1对内存的响应是不一样的。 从堆内存变化来看, JDK11下G1更加倾向于尽可能的利用内存,不着急回收。 而JDK8下G1则是倾向于尽可能的先回收内存。 从图中看,JDK8下G1的实际使用的堆内存大小基本是JDK11下G1的一半

    小结

    如果代码保持不变,但是JVM参数中设置Xms和Xmx相同的话,不管是否有FullGC,堆内存大小都不发生变化,也就不释放内存给操作系统

    GC后如何归还内存给操作系统:

    • 能不能归还,主要依赖于Xms和Xmx是否相等
    • 何时归还,主要依赖于JDK版本和垃圾回收器类型

    只有FullGC的时候才能真正触发堆内存收缩归还OS。YGC是不能使JVM主动归还内存给操作系统的

    尽量保持Xms和Xmx一致,这样可以减少堆内存调整带来的性能损耗,也可以减少堆内存调整带来的无内存风险

    参考:

    https://segmentfault.com/a/1190000019856974

    https://www.cnblogs.com/androidsuperman/p/11743103.html

    http://blog.dutycode.com/archives/jvmjvm%E7%9A%84xms%E5%8F%82%E6%95%B0%E5%92%8Clinuxtop%E5%91%BD%E4%BB%A4%E7%9A%84res%E5%85%B3%E7%B3%BB

  • 相关阅读:
    十天学前端之JS篇(六)
    Tomcat 源码分析 (Digester类的使用) (十)
    机器人C++库(9)Robotics Library 之机器人建模(VRML)、工作场景Scances建模(VRML)
    Elasticsearch入门教程(未完成)
    MATLAB算法实战应用案例精讲-【优化算法】受精优化算法(FO)(附MATLAB代码实现)
    NodeJs 性能提升方案
    34、CSS进阶——行高的取值以及常见的长度单位
    华为云云耀云服务器L实例评测 | 实例使用教学之综合导览
    【JavaScript】js判断一个变量是数组
    计算机组成原理-输入输出系统(持续更新中)
  • 原文地址:https://blog.csdn.net/m0_67401499/article/details/126744568