深入理解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的次数和影响,从而为应用程序提供更平滑的性能体验。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/197401.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

STM32串口接收不定长数据(空闲中断+DMA)

玩转 STM32 单片机,肯定离不开串口。串口使用一个称为串行通信协议的协议来管理数据传输,该协议在数据传输期间控制数据流,包括数据位数、波特率、校验位和停止位等。由于串口简单易用,在各种产品交互中都有广泛应用。 但在使用串…

Java中的锁

Java常见锁 【超全面】_java锁-CSDN博客 一文看懂Java中的锁 - 知乎 一文读懂 Java 中的各种锁 - 知乎 java中锁的分类总结_java string 锁-CSDN博客 https://www.cnblogs.com/hangychn/p/17184814.html https://www.cnblogs.com/durenniu/p/10949491.html 锁-Java - 知…

【LeeCode】链表总结

理论基础 链表的种类主要为:单链表,双链表,循环链表 链表的存储方式:链表的节点在内存中是分散存储的,通过指针连在一起。 链表是如何进行增删改查的。 数组和链表在不同场景下的性能分析。 经典题目 虚拟头节点 …

Passkey模式

以下内容参考自谷歌的代码: 原文地址 Passkey 模式用于以比简单地将其他类设置为友元类更细粒度的方式将类的部分方法暴露给另一个类。本质上,它涉及创建一个“passkey”类,该类只能由特定的其他类构造,并要求在调用您希望限制使…

手把手教你做基于stm32的红外、语音、按键智能灯光控制(下)

目录: 4.6. DHT11温湿度传感器模块4.7. 语音识别模块4.7. OLED显示屏模块 5. 不同的工作模式6. 总结 在接着(上)写之前,首先来看一下效果: 链接: link 需要源码什么的可以私信我哦 4.6. DHT11温湿度传感器模块 这个…

C陷阱与缺陷——第2章语法陷阱

1. 理解函数声明 硬件将调用首地址为0位置的子例程 (*(void(*)())0)(); 任何C变量的声明都由两部分组成:类型以及一组类似表达式的声明符,声明符从表面看与表达式有些类似,对它求值应该返回一个声明中给定类型的结果。 假定变量fp是一个函…

[go 面试] 构建高效微服务通信:选择合适的通信方式

关注公众号【爱发白日梦的后端】分享技术干货、读书笔记、开源项目、实战经验、高效开发工具等,您的关注将是我的更新动力! 构建分布式系统或微服务架构时,服务间通信成为至关重要的一环。不同的通信方式各有优劣,因此在选择时需根…

数据库sql是什么?

数据库是什么? 为了将数据持久保存,java中的对象都是在内存中,程序结束数据就会销毁,在学习IO过程中将数据存储在文件中,但是内容整体是字符串,使用起来不方便。所以需要数据库长久保存数据还便于提取 数…

值和引用的传递与初始化块

值:1.基本类型数据;2.对象的地址 引用:传句柄的地址 在java中只有按值传递没有按引用传递,c语言也只有按值传递,只有c才有引用传递。 块: 静态块>块>构造方法 package com;public class Person {pri…

揭秘预付费电表怎么无线收费——方便快捷收费

【摘要】针对目前市场上普遍以Ic卡作为售电介质的预付费售电系统存在的问题,介绍了一种新型的无线预付费售电系统及其构成,并给出了整个系统设计的完整方案。整个系统包括用户终端和电力管理系统端,它们之间通过双工通信可以将用户用电信息和…

Kubernetes存储搭建NFS挂载失败处理

搞NFS存储时候发现如下问题: Events:Type Reason Age From Message---- ------ ---- ---- -------Normal Scheduled 5m1s default-scheduler Successful…

JavaScrip之函数柯理化、参数复用、提前确认、延迟执行

MENU 1、参数复用2、提前确认 (惰性函数)3、延迟运行4、初步封装5、递归封装6、经典面试题 1、参数复用 // 普通函数验证 function check(regExp, text) {return regExp.test(text); }console.log(check(/^\d$/g, 123)); // true console.log(check(/^\d$/g, 2d)); // false…

LeetCode算法练习top100:(7)递归回溯

package top100.递归回溯;import java.util.*;public class TOP {//46. 不含重复数字的全排列List<List<Integer>> res new ArrayList<>();public List<List<Integer>> permute(int[] nums) {LinkedList<Integer> path new LinkedList&l…

电子学会全国青少年软件编程等级考试 中小学生python一级历年真题解析【更新至2023年9月 持续更新】

中国电子学会python等级考试一级历年真题解析 一、考级知识点分析 一、 了解Python多种开发环境&#xff0c;熟练使用Python自带的IDLE开 发环境&#xff0c;能够进行程序编写、调试和分析&#xff0c;具备使用Python开发 环境进行程序设计的能力 了解Python常见的几种编程环…

堆排序详细解读

简介 堆排序是一种基于二叉堆数据结构的排序算法&#xff0c;它的特点是不同于传统的比较排序算法&#xff0c;它是通过建立一个堆结构来实现的。堆排序分为两个阶段&#xff0c;首先建立堆&#xff0c;然后逐步将堆顶元素与堆的最后一个元素交换并调整堆&#xff0c;使得最大…

EM32DX-C2【C#】

1说明&#xff1a; 分布式io&#xff0c;CAN总线&#xff0c;C#上位机二次开发&#xff08;usb转CAN模块&#xff09; 2DI&#xff1a; 公共端是&#xff1a; 0V【GND】 X0~X15&#xff1a;自带24v 寄存器地址&#xff1a;0x6100-01 6100H DI输入寄存器 16-bit &#x…

ROS2 galactic生成的bag包里的MarkerArray在humble下播放不正常

近期发现ROS2 galactic下生成的bag包在humble下回放时使用rviz可视化&#xff0c;bag里的点云可以正常看到&#xff0c;但是使用Marker和MarkerArray画的box却死活看不到&#xff0c;感觉很纳闷&#xff0c;看网上有人报告说foxy下生成的bag包在galactic下播放会报SQL错误&…

Redis部署-哨兵模式

目录 redis sentinel相关名词 redis sentinel架构 故障转移流程 基于docker搭建redis哨兵 准备工作 搭建过程 模拟主节点宕机,观察哨兵节点的工作流程 哨兵重新选取主节点的流程 1.主观下线 2.客观下线 3.哨兵节点推举出一个leader节点 4.leader选举完毕,leader挑选…

RflySim | 姿态控制器设计实验一

姿态控制器设计实验1 一. 姿态控制设计简介 本文是建立在多旋翼的姿态即控制器中的反馈信号能够被较好地估计的前提下&#xff0c;控制器中的反馈信号是估计值。不过&#xff0c;为了更加简便根据分离原理&#xff0c;我们用真值代替反馈信号。本文的目的是让多旋翼的姿态能够…

Linux入门攻坚——7、磁盘管理——文件系统挂载管理及RAID、LVM

已经安装文件系统的分区需要经过挂载才能使用。 一切文件系统的使用都是从根开始&#xff0c;根是文件系统的起始点。 计算机启动过程&#xff1a;加电自检——bootloader——kernel——rootfs——/sbin/init kernel第一步要加载根系统。 将额外文件系统与根文件系统某现存的…