深入理解JVM内存空间的担保策略
Java虚拟机(JVM)的内存管理是Java性能调优中最重要的方面之一,特别是在处理大型应用和服务时。JVM内存管理的一个关键组成部分是垃圾回收(GC)。在GC过程中,JVM需要确保有足够的内存来创建新对象,同时还要清理不再使用的对象。而空间担保策略是JVM为了应对这一需求而采取的一种内部机制。本文将深入探讨JVM的空间担保策略是什么,以及它是如何工作的。
什么是JVM空间担保策略?
空间担保策略(Promotion Guarantee)是JVM中的一种机制,确保在Minor GC时,存活的对象能够成功晋升到老年代。如果老年代没有足够的空间来接收新晋升的对象,JVM可能会提前触发一次Full GC来释放空间,或者调整自己的内存分配策略以避免此类情况的发生。
JVM内存结构
为了理解空间担保策略,我们必须首先了解JVM内存的结构。JVM内存主要分为几个区域:
- 新生代(Young Generation):新创建的对象首先被放置在新生代。新生代包括一个Eden区和两个Survivor区(通常称为S0和S1)。
- 老年代(Old Generation): 存活经过一定次数GC的对象会被晋升到老年代。
- 元空间(Metaspace): 用于存放类元数据的区域,替代了早期版本的Java中的永久代(PermGen)。
空间担保的工作原理
- jdk6以前
在进行Minor GC前,虚拟机会检查老年代最大可用的连续空间是否大于新生代所有对象总空间。如果这个条件不能满足,虚拟机会查看 -XX:HandlePromotionFailure
设置是否允许担保失败。如果不允许(false),那么会提前进行一次Full GC来清理老年代并为新生代晋升的对象腾出空间。如果允许担保失败(true),那么只要老年代剩余空间大于历次晋升到老年代对象的平均大小即可进行Minor GC,否则也要提前进行Full GC。
JDK6源码
- jdk6以后
从JDK 7开始,HotSpot虚拟机的垃圾收集器在做Minor GC之前的空间分配担保策略上进行了调整,取消了之前版本中的 -XX:HandlePromotionFailure 选项
。每次都会判断老年代剩余最大连续空间大于历次Minor GC晋升的平均大小
或者 大于新生代所有对象的大小总和
, 大于任意一个,就允许触发MinorGC,反之触发 Full GC
JDK8源码
举例说明空间担保策略
假设一个Java应用配置了 -Xmx100M -Xms100M -XX:+UseSerialGC 来设置使用串行垃圾回收器和100MB堆内存,且无其他特别的内存区域大小参数设置。
在这种情况下,JVM会分配一定比例的内存给新生代和老年代。如果运行过程中发现老年代的连续空间小于新生代中所有对象的总空间,JVM会进行Full GC而不是Minor GC,这是为了防止在Minor GC过程中因为老年代空间不足而导致GC失败。
假设在一次Minor GC后,有2MB的对象需要晋升到老年代,而老年代的连续可用空间只有1MB,并且小于历次Minor GC晋升的平均大小。根据空间担保策略,JVM将执行Full GC。如果Full GC之后老年代的可用空间仍然无法满足晋升需求,JVM会抛出 OutOfMemoryError
。
而在实践中,有时候会关闭担保失败 -XX:-HandlePromotionFailure
,在早期版本的HotSpot虚拟机中,默认是开启的,但在JDK 7及其之后的版本中,这个选项已经被移除了
,因为JVM的垃圾收集器已经被优化到即使在非常紧张的内存情况下也可以很好地进行垃圾回收。
如何调优空间担保策略?
空间担保策略的调优通常涉及几个关键的JVM参数:
- -XX:SurvivorRatio: 设置新生代中Eden区与Survivor区的比例。
- -XX:NewRatio: 设置新生代与老年代的比例。
- -XX:MaxTenuringThreshold: 设置对象在新生代的存活次数,超过这个次数的对象会被晋升到老年代。
- -XX:PretenureSizeThreshold: 设置大小阈值,超过这个大小的对象不会在新生代分配,而直接在老年代分配。
调优通常需要根据应用程序的具体情况来进行。监控工具(如jstat, VisualVM或其他商业监控工具)可以帮助你理解内存使用情况,并据此做出调整。
总结
如果没有空间担保,Minor GC会进行尝试,很可能在晋升过程中失败,因为老年代没有足够的空间。这时JVM可能会抛出 OutOfMemoryError
,或者尝试一次昂贵的Full GC
来强制回收空间。
而开启空间担保策略,JVM在开始Minor GC之前会检查老年代是否有足够的空间。在这个情况下,JVM会认识到老年代空间不足,因此可能直接触发Full GC,来确保不会在Minor GC过程中出现内存分配失败
。
总之,空间担保策略是一种预防措施,保障JVM在进行Minor GC时的内存分配安全性,尽量减少Full GC的发生,以提高系统的性能和稳定性。
注:每次的垃圾回收都是对系统资源的一次消耗,因此适当的调优可以减少GC的次数和影响,从而为应用程序提供更平滑的性能体验。
相关文章:

Java中volatile 的使用场景有哪些?
volatile是一种轻量级的同步机制,它能保证共享变量的可见性,同时禁止重排序保证了操作的有序性,但是它无法保证原子性。所以使用volatilevolatile。

JVM优雅退出
在某个Java应用增加新功能,缩容机器,或者应用以及机器发生异常,通常会停止正在运行的应用,该应用通常正在运行着任务,如果停止应用的操作处理不当的话,很有可能会导致数据丢失,损坏,从而影响业务。所以在停止应用的时候,需要考虑如何安全优雅的退出。维护了所有已经注册的钩子,由于jvm本身没有提供好用的方法去移除已经注册的钩子,可以通过反射的方式调用。对于强制关闭的几种情况,会直接停止JVM进程,JVM不会调用已注册的。对于正常关闭、异常关闭的几种情况,JVM关闭前,都会调用已注册的。

10万字总结Java8到21新特性详解
Java 8 是Java历史上一个重大的版本更新,发布于2014年3月18日。Lambda 表达式是在 Java 8 中引入,并且被吹捧为 Java 8 最大的特性。它是函数式编程的的一个重要特性,标志着 Java 向函数式编程迈出了重要的第一步。或者其中parameters:是 Lambda表达式的参数列表,可以为空或包含一个或多个参数。->:是 Lambda 操作符,用于将参数和 Lambda 主体分开。expression:是 Lambda 表达式的返回值,或者在主体中执行的单一表达式。

垃圾回收器ZGC应用分析总结
ZGC 是一款低延迟、高吞吐的垃圾回收器,由 Oracle 公司开发。它适用于大型、多核、内存容量较大的应用程序。ZGC 的设计目标是在最大限度地减少停顿时间的同时,为大型内存提供可伸缩性,并为生产部署提供高吞吐量和稳定性。它的目标是以不到 10 毫秒的暂停时间来控制 100MB 到 4TB 的内存。此外,ZGC 还致力于避免全局 JVM 暂停,从而提高系统的可用性。简单来说,它的设计目标是在不超过 10 毫秒的暂停时间内,尽可能地回收大量的堆内存。低延迟:ZGC 的主要目标是最小化 GC 暂停时间。

JVM安全退出(如何优雅的关闭java服务)
为了保障应用重启过程中异步操作的执行,避免强制退出JVM可能产生的各种问题,我们可以采用关闭钩子、自定义信号的方式,主动的通知JVM退出,并在JVM关闭前,执行应用程序的一些扫尾工作,进一步保证应用程序可以安全的退出。

Java中的四种访问权限(private,public,protected,无修饰)
/实体类属性和数据库字段名称不一致//实体类属性和数据库字段名称不一致return id;return age;emp.test();//直接调用public修饰的变量//private修饰的变量进行赋值//调用private修饰的变量1、public修饰符定义的属性和方法通过对象实例化进行调用,2、private修饰的属性通过set、get方法进行调用。

java并发编程八 CAS 与 volatile和原子整数与原子引用
它可以用来修饰成员变量和静态成员变量,他可以避免线程从自己的工作缓存中查找变量的值,必须到主存中获取它的值,线程操作 volatile 变量都是直接操作主存。在java并发编程七六中,可以看到的 AtomicInteger 的解决方法,内部并没有用锁来保护共享变量的线程安全。那么它是如何实现的呢?其中的关键是 compareAndSet,它的简称就是 CAS (也有 Compare And Swap 的说法),它必须是原子操作。获取共享变量时,为了保证该变量的可见性,需要使用 volatile 修饰。