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

前言

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

测试代码

public class MemoryRecycleTest {

    static volatile List<OOMobject> 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<OOMobject> 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("");
    }
}

JDK8 CMS

JVM参数

-Xms128M -Xmx2048M -XX:+UseConcMarkSweepGC

控制台打印的内容

第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

VisualVM监控的堆内存情况

在这里插入图片描述

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

JDK8 G1

JVM参数

-Xms128M -Xmx2048M -XX:+UseG1GC

VisualVM监控的堆内存情况

在这里插入图片描述

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

JDK11 CMS

JVM参数

-Xms128M -Xmx2048M -XX:+UseConcMarkSweepGC

VisualVM监控的堆内存情况

在这里插入图片描述

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

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

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

VisualVM监控的堆内存情况

在这里插入图片描述

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

JDK11 G1

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

JVM参数

-Xms128M -Xmx2048M

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

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值