当前位置: 首页 > news >正文

Java底层堆内存、GC等知识点阐述

Java底层

  • 堆(年轻代与老年代知识点)
    • 年轻代(新生区) 堆
      • 堆≠伊甸园
      • 堆(含有三个区域:伊甸园和两个幸存者)
      • 为什么是两个幸存者,一个幸存者存在不就可以嘛?
      • 原因分析
      • 分析总结
    • 老年代 堆
    • GC种类
    • 垃圾回收
      • 什么是垃圾回收
      • 为什么需要垃圾回收
      • 回收器
      • 新生代垃圾回收器
      • 老年代垃圾回收器
      • JDK1.8默认的垃圾回收器,以下为jdk1.8说明
    • springboot GC垃圾回收器使用示例

堆(年轻代与老年代知识点)

年轻代(新生区) 堆

新生代主要用于存放新创建的对象,这些对象由于频繁创建和销毁,因此会频繁触发Minor GC(小型垃圾回收)来清理不再使用的对象,以保持新生代的健康和高效运行。

堆≠伊甸园

在Java虚拟机(JVM)中,内存被划分为不同的区域其中堆是JVM管理的最大一块内存空间,用于存放对象实例和数组。堆内存的大小是可以调节的,并且所有的线程共享Java堆。然而,伊甸园(Eden space)是堆内存中的一个特定区域,位于新生代中,是对象出生的地方,所有新的对象都是在伊甸园区被创建的新生代还包括幸存区(Survivor space),分为两个幸存区,用于存放从伊甸园中幸存下来的对象。这些区域共同构成了堆内存的一部分,但伊甸园只是堆内存中的一个子区域,专门用于对象的初始创建‌
简而言之,堆是Java内存管理的核心区域,而伊甸园是堆内存中的一个具体区域,专门用于新生对象的创建。如下图所示1-1
在这里插入图片描述
(图1-1)

堆(含有三个区域:伊甸园和两个幸存者)

当伊甸园区满了以后会使用GC回收不在引用对象销毁,而剩余对象在两个幸存者之间倒腾(折腾)。

为什么是两个幸存者,一个幸存者存在不就可以嘛?

原因分析

主要原因是内存碎片,内存碎片分为内部和外部碎片,下面以外部碎片举例
1.现在创建了4个对象 ,分别是stu(学生)、clas(年级)、grades(成绩)、sub(科目)
在这里插入图片描述
2.现在垃圾收集器,从内存中删除了grades(成绩),内存就变成了如下结构
在这里插入图片描述
3.假如现在需要创建一个更大的新的“tea(老师)”对象,但是在图中可以看出,tea没有足够大的连续空间可以放了,那么由此造成空间浪费,成为内存碎片
在这里插入图片描述
4.FULL GC
当产生内存碎片时,会调用 full gc 来整理内存,就可以将tea放入了
在这里插入图片描述

分析总结

full gc 最终会导致JVM暂停较长时间,非常影响效率。因此为了解决存在碎片带来的麻烦,再划分一个幸存者区,将gc回收之后的Survivor0区(from)倒腾到Survivor1区(幸存者区1或称为To区),就可以避免这种问题,是一种空间换时间的思路。

老年代 堆

Java中的老年代(Old Generation)是Java虚拟机(JVM)内存管理中的一个重要部分,它和新生代一起共同管理Java对象的内存分配和回收

对象在新生代中存活了一定的时间(通常是通过复制算法来判断对象的“年龄”),并且没有被回收,那么这些对象就会被“晋升”到老年代。老年代中的对象相对稳定,因此Major GC(大型垃圾回收)不会像Minor GC那样频繁执行。老年代的对象比较稳定,所以在进行Major GC前一般都先进行了一次Minor GC,使得有新生代的对象晋身入老年代,导致空间不够用时才触发Major GC进行垃圾回收。当老年代也满了装不下的时候,就会抛出OOM(Out of Memory)内存溢出异常。

老年代的垃圾回收(Major GC)采用标记-清除算法,首先扫描所有老年代中的对象,标记出存活的对象,然后回收没有标记的对象。这种方式的耗时比较长,因为需要扫描并回收大量对象,可能会导致内存碎片化。为了减少内存损耗,可能需要进行合并或标记出空闲空间以便下次直接分配。

总的来说,老年代是Java内存管理中的一个关键部分,它存储了那些在新生代中存活下来并且不再频繁变动的对象,这些对象由于相对稳定,因此其垃圾回收的频率要比新生代低得多‌

GC种类

1.Minor GC‌:主要发生在新生代(Young Generation),包括Eden区和Survivor区。当Eden区满了或者Survivor区无法容纳从Eden区复制过来的对象时,会触发Minor GC。Minor GC主要负责清理年轻代中的内存,包括Eden区和Survivor区的清理‌12。
‌2.Major GC‌:当老年代(Old Generation)内存不足时,会触发Major GC,主要负责清理老年代区的内存‌12。
‌3.Full GC‌:Full GC是清理整个堆空间的垃圾回收,包括年轻代和老年代。Full GC通常是由于Minor GC和Major GC频繁触发,或者系统资源紧张等原因导致的‌

垃圾回收

什么是垃圾回收

在Java中,对象是在堆内存上分配的,当这些对象不再被任何部分的应用所引用时,这些对象就被认为是垃圾。垃圾回收器的任务就是找到这些不再被需要的对象,并释放它们占据的内存空间,以供新对象使用。

为什么需要垃圾回收

无法控制内存是许多程序可能遇到的问题。例如,内存泄漏,这通常发生在程序不再需要的数据没有被及时清理的情况下,长此以往可能导致内存耗尽,最终程序崩溃。通过自动垃圾回收,Java试图避免这种情况。

回收器

垃圾回收器可以根据具体的业务场景选择最合适的进行搭配使用。例如,新生代收集器可以与老年代收集器搭配使用,如Serial与Serial Old搭配,或ParNew与CMS搭配,以满足不同的性能和资源需求‌

新生代垃圾回收器

‌新生代的垃圾回收器包括Serial收集器、ParNew收集器、Parallel Scavenge收集器。

Serial收集器:单线程进行垃圾回收操作,适用于单核服务器或Client模式(桌面应用)。

# 添加该参数来显式的使用Serial垃圾收集器。
-XX:+UseSerialGC

ParNew收集器:Serial收集器的多线程版本,适用于多核CPU环境,多线程进行新生代的垃圾回收。

# 指定使用CMS后,会默认使用ParNew作为新生代收集器。
-XX:+UseConcMarkSweepGC
# 强制指定使用ParNew。   
-XX:+UseParNewGC
# 指定垃圾收集的线程数量,ParNew默认开启的收集线程与CPU的数量相同。
-XX:ParallelGCThreads

Parallel Scavenge收集器:适用于追求吞吐量高的场景,当Eden区内存不足时,发生Minor GC,采用标记复制法进行垃圾回收。

# Parallel Scavenge收集器提供了两个参数来用于精确控制吞吐量,
#一是控制最大垃圾收集停顿时间的 -XX:MaxGCPauseMillis参数,
#二是控制吞吐量大小的 -XX:GCTimeRatio参数;#参数允许的值是一个大于0的毫秒数,收集器将尽可能的保证内存垃圾回收花费的时间不超过设定的值(但是,并不是越小越好,GC停顿时间缩短是以牺牲吞吐量和新生代空间来换取的,如果设置的值太小,将会导致频繁GC,这样虽然GC停顿时间下来了,但是吞吐量也下来了)。
-XX:MaxGCPauseMillis# 参数的值是一个大于0且小于100的整数,也就是垃圾收集时间占总时间的比率,默认值是99,就是允许最大1%(即1/(1+99))的垃圾收集时间。
-XX:GCTimeRatio#参数是一个开关,如果这个参数打开之后,虚拟机会根据当前系统运行情况收集监控信息,动态调整新生代的比例、老年大大小等细节参数,以提供最合适的停顿时间或最大的吞吐量,这种调节方式称为GC自适应的调节策略。
-XX:UseAdaptiveSizePolicy

老年代垃圾回收器

老年代的垃圾回收器包括Serial Old收集器、CMS收集器、Parallel Old收集器。

Serial Old收集器:用于老年代的垃圾回收,采用单线程方式进行回收操作。
CMS收集器:以最短回收停顿时间为目标的垃圾回收器,采用并发标记和清除的方式进行老年代的垃圾回收。

1.初始标记(CMS initial mark):仅仅只是标记一下GC Roots能直接关联到的对象,速度很快,需要“Stop The World”。
2.并发标记(CMS concurrent mark):进行GC Roots Tracing的过程,在整个过程中耗时最长。
3.重新标记(CMS remark):为了修正并发标记期间因用户程序继续运作而导致标记产生变动的那一部分对象的标记记录,这个阶段的停顿时间一般会比初始标记阶段稍长一些,但远比并发标记的时间短。此阶段也需要“Stop The World”。
4.并发清除(CMS concurrent sweep)由于整个过程中耗时最长的并发标记和并发清除过程收集器线程都可以与用户线程一起工作
## 新生代 ParNew + 老年代 CMS + 老年代 Serial Old
# 某些版本的参数是这样的: -XX:+UseConcurrentMarkSweepGC
-XX:+UseConcMarkSweepGC# 响应时间优先,停顿时间,是一个建议时间,GC会尝试用各种手段达到这个时间,比如减小年轻代,默认 18446744073709551615  
-XX:MaxGCPauseMillis
# 吞吐量优先,设置JVM吞吐量要达到的目标值, GC时间占用程序运行时间的百分比的差值,默认是 99
# 也就应用程序线程应该运行至少99%的总执行时间,GC占 1%
-XX:GCTimeRatio=99# 并行收集器(ParNew , STW,  YGC)的线程数,默认CPU所支持的线程数,如果CPU所支持的线程数大于8,则 默认 8 + (logical_processor -8)*(5/8)
-XX:+ParallelGCThreads
#  CMS垃圾回收线程数量
-XX:ParallelCMSThreads# 解决 CMS `Memory Fragmentation` 碎片化, 开启FGC时进行压缩,以及多少次FGC之后进行压缩
-XX:+UseCMSCompactAtFullCollection
-XX:CMSFullGCsBeforeCompaction=3
# 解决 CMS `Concurrent mode failure` ,`Promotion Failed`晋升失败  
# 使用多少比例的老年代后开始CMS收集,默认是68%(近似值),如果频繁发生SerialOld卡顿,应该调小,(频繁CMS回收)
-XX:+UseCMSInitiatingOccupancyOnly
-XX:CMSInitiatingOccupancyFraction=70# 开启 CMS 元空间的垃圾回收
-XX:+CMSClassUnloadingEnabled
# -XX:CMSInitiatingPermOccupancyFraction  (JDK8已经移除)

Parallel Old收集器:并行老年代垃圾回收器,使用多线程并行进行老年代的垃圾回收。

JDK1.8默认的垃圾回收器,以下为jdk1.8说明

默认:Parallel GC(Throughput GC)
Parallel是一种多线程的垃圾回收器,他主要的目标就是最大化吞吐量(吞吐量是值应用程序实际执行代码的时间相对于总时间(包括垃圾回收时间))的比例。Parallel GC在老年代和新生代垃圾回收会使用多线程执行。

新生代:使用的是复制算法。

老年代:采用的是MSC回收算法

MSC (Mark-Sweep-Compact)
Marking 标记
在标记就单,垃圾回收器回遍历堆中的所有对象,识别出来哪些对象是可达的。并且将这些活跃的对象标记为活跃的伙计。这个过程通常是通过(GC Roots,通常是静态变量、栈中的引用等)开始进行的。

Sweeping 清除
在清除阶段,垃圾回收器回扫描堆中的没有被标记的对象,将这些对象从内存中释放,一边以后可以分配给新的对象。这一步只会简单清楚不可达的对象,不会对内存进行整理。因此可能导致内存碎片化的问题。

Compacting 压缩
压缩阶段可以解决内存碎片化的问题。在清除阶段之后,堆中可能出现碎片,这些碎片会导致内存分配不连续和低效。

工作流程
标记阶段:首先,标记所有存活的对象。

清除阶段:接着,清除所有未标记的对象,将它们的内存回收。

压缩阶段:最后,压缩存活的对象,将它们集中到堆的一端,释放出连续的可用内存。

优点
可以有效地回收内存并减少内存碎片,然后后续的内存分配更加高效。

适合老年代,因为老年代中对象存活率较高。内存碎片可能更为严重。

缺点
MSC可能会导致比较长时间的Stop-the-world(STW)停顿。因为压缩过程中,整个堆中的所有对象都可能需要移动。

开销比较大:压缩阶段需要复制和移动大量的对象,这增加了垃圾回收的开销。

springboot GC垃圾回收器使用示例

方式一、在application.properties或application.yml文件中设置JVM参数

# 设置JVM参数
spring.jmx.enabled=false# 示例使用并行垃圾收集器
spring.application.jvm-args=-XX:+UseParallelGC

方式二、在启动Spring Boot应用时通过命令行指定

java -XX:+UseParallelGC -jar your-spring-boot-app.jar
使用串行垃圾收集器:-XX:+UseSerialGC使用并行垃圾收集器:-XX:+UseParallelGC使用并行垃圾收集器(G1):-XX:+UseG1GC设置新生代的并行线程数:-XX:ParallelGCThreads=<number>设置最大垃圾收集暂停时间(仅G1):-XX:MaxGCPauseMillis=<milliseconds>设置G1的区域数量:-XX:G1HeapRegionSize=<size>

在这里插入图片描述

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • 8、引用
  • 【机器学习】CNN在计算机视觉中的应用
  • PowerShell脚本编写:自动化Windows开发工作流程
  • Python(PyTorch)物理变化可微分神经算法
  • 北京博科测试
  • Java并发编程12
  • es 7.17.23安装ik插件启动失败,access denied,Permission
  • uniapp开发微信小程序调用微信支付
  • 宠物空气净化器吸猫毛有用吗?希喂、美的、霍尼韦尔三款亲测分享
  • C++类和对象(3)——this指针
  • Redis中的缓存穿透、缓存击穿、缓存雪崩(面试版)
  • matlab 创建复数数组
  • SQL典型练习题
  • 云渲染解决:笔记本渲染很伤电脑吗?如何保护你的电脑?
  • 【计算机组成原理】2.2.6 数据的存储和排列
  • 收藏网友的 源程序下载网
  • [deviceone开发]-do_Webview的基本示例
  • [译]前端离线指南(上)
  • 【挥舞JS】JS实现继承,封装一个extends方法
  • 【跃迁之路】【733天】程序员高效学习方法论探索系列(实验阶段490-2019.2.23)...
  • docker python 配置
  • Docker入门(二) - Dockerfile
  • ES6之路之模块详解
  • JavaScript类型识别
  • javascript数组去重/查找/插入/删除
  • Laravel Mix运行时关于es2015报错解决方案
  • Mac转Windows的拯救指南
  • OpenStack安装流程(juno版)- 添加网络服务(neutron)- controller节点
  • PHP CLI应用的调试原理
  • swift基础之_对象 实例方法 对象方法。
  • Vue.js 移动端适配之 vw 解决方案
  • 从零开始的webpack生活-0x009:FilesLoader装载文件
  • 猫头鹰的深夜翻译:JDK9 NotNullOrElse方法
  • 学习ES6 变量的解构赋值
  • 《TCP IP 详解卷1:协议》阅读笔记 - 第六章
  • ​力扣解法汇总946-验证栈序列
  • #HarmonyOS:软件安装window和mac预览Hello World
  • #Ubuntu(修改root信息)
  • #我与Java虚拟机的故事#连载02:“小蓝”陪伴的日日夜夜
  • #我与Java虚拟机的故事#连载17:我的Java技术水平有了一个本质的提升
  • (02)Cartographer源码无死角解析-(03) 新数据运行与地图保存、加载地图启动仅定位模式
  • (C#)获取字符编码的类
  • (C语言)编写程序将一个4×4的数组进行顺时针旋转90度后输出。
  • (Python) SOAP Web Service (HTTP POST)
  • (第一天)包装对象、作用域、创建对象
  • (二)十分简易快速 自己训练样本 opencv级联lbp分类器 车牌识别
  • (附源码)SSM环卫人员管理平台 计算机毕设36412
  • (免费领源码)Java#ssm#MySQL 创意商城03663-计算机毕业设计项目选题推荐
  • (深度全面解析)ChatGPT的重大更新给创业者带来了哪些红利机会
  • (十七)Flask之大型项目目录结构示例【二扣蓝图】
  • (详细版)Vary: Scaling up the Vision Vocabulary for Large Vision-Language Models
  • (一)【Jmeter】JDK及Jmeter的安装部署及简单配置
  • (转)创业家杂志:UCWEB天使第一步
  • .java 9 找不到符号_java找不到符号
  • .NET Core中的去虚