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

彻底搞懂 Java 垃圾回收(GC)

在 Java 后端开发、面试、线上性能优化、OOM 排查中GC垃圾回收都是绕不开的核心基石。很多人只知道 GC 是自动回收内存但到底怎么回收、什么时候回收、为什么会卡顿、不同回收器区别是什么一知半解。这篇文章我用最通俗、最详细、最贴近实战的方式把 GC 从头到尾讲透。一、什么是 GC为什么 Java 必须要有 GC1.1 GC 是什么GC Garbage Collection垃圾回收。简单说JVM 自动帮我们找到内存中 “不再使用的对象”并释放内存避免内存泄漏和 OOM。1.2 为什么需要 GCC/C 需要手动malloc/free、new/delete忘记释放 → 内存泄漏重复释放 → 程序崩溃大项目根本管不住Java 的核心思想之一就是让开发者专注业务逻辑内存管理交给 JVM。GC 就是负责这件事的核心机制。二、GC 要回答的三个核心问题所有 GC 机制都围绕这三句话展开哪些对象是垃圾判定算法什么时候回收触发时机怎么回收最高效、最低卡顿回收算法 回收器下面我们逐个彻底讲清楚。三、如何判断一个对象是 “垃圾”3.1 引用计数法简单但被淘汰原理对象被引用一次计数器 1引用失效计数器 -1计数器 0 → 判定为垃圾优点简单、实时。致命缺点无法解决循环引用A a new A(); B b new B(); a.b b; b.a a; a null; b null;此时计数器都不为 0 →永远无法被回收导致内存泄漏。所以JVM 没有使用引用计数。3.2 可达性分析算法JVM 真正使用原理从GC Roots根节点开始向下搜索能搜索到达的对象 存活无法到达 垃圾。哪些可以作为 GC Roots虚拟机栈中引用的对象局部变量、方法参数类静态属性引用的对象常量引用的对象本地方法栈Native 方法引用的对象同步锁synchronized持有的对象只要不在 GC Roots 引用链上就会被判定为可回收对象。3.3 对象真正死亡的过程第一次可达性分析标记为可回收判断是否有必要执行 finalize ()如果在 finalize () 中重新被引用 → “自救” 成功否则第二次标记 →真正成为垃圾注意finalize () 不推荐使用运行代价高不确定性大。四、Java 中的 4 种引用类型面试高频 实战重要不同引用决定了 GC什么时候回收对象。1. 强引用StrongObject obj new Object();最常见只要强引用还在GC 永远不会回收内存不足时宁可 OOM 也不回收2. 软引用Soft内存足够不回收内存即将 OOM才回收适合缓存SoftReferenceObject sr new SoftReference(obj);3. 弱引用Weak只要发生 GC立即回收生命周期最多活到下一次 GC适合短周期缓存、防止内存泄漏WeakReferenceObject wr new WeakReference(obj);4. 虚引用Phantom完全不影响生命周期唯一作用对象被回收时收到通知用来管理直接内存NIO五、垃圾回收的三种基础算法5.1 标记 - 清除算法步骤标记所有垃圾对象统一清除优点简单。缺点效率不高产生大量内存碎片→ 大对象无法分配空间提前触发 Full GC5.2 复制算法把内存分成两块From 和 To只在 From 分配对象垃圾回收时将存活对象复制到 To清空整个 From交换 From 和 To 角色优点速度极快无内存碎片缺点浪费 50% 内存适用场景新生代对象存活率极低5.3 标记 - 整理算法结合前两者优点标记垃圾让所有存活对象向一端移动清理边界以外的内存优点无内存碎片不浪费内存缺点效率较慢适用场景老年代对象存活率高六、Java 分代回收模型JVM GC 的灵魂JVM 根据对象存活时间把堆内存分成两块新生代 老年代不同区域使用不同回收算法这就是分代回收。6.1 新生代Young Generation特点对象朝生夕死每次 GC 90% 对象死亡使用复制算法内部结构Eden 区新对象出生的地方S0From SurvivorS1To Survivor对象分配 回收流程新对象 → EdenEden 满 →Minor GC存活对象复制到 S0下次 GCEden S0 存活 → S1交换 S0/S1对象在 Survivor 来回躲避15 次 GC→ 进入老年代年龄阈值由-XX:MaxTenuringThreshold控制6.2 老年代Old Generation特点对象存活时间极长存活率高使用标记 - 清除 或 标记 - 整理触发 GCMajor GC老年代回收Full GC整堆回收新生代 老年代 元空间速度慢、会 STW线上要尽量避免。6.3 元空间 MetaspaceJDK 8 以后取代永久代存放类信息、方法、常量、代理类不属于堆满了也会触发 GC七、Stop The WorldSTW—— GC 最关键概念STW Stop The World垃圾回收时所有用户业务线程全部暂停直到 GC 结束。所有 GC 都无法完全避免 STW只能尽量缩短。STW 时间越长 → 接口越慢、服务越卡低延迟应用支付、游戏、API必须严控 STW八、7 大经典垃圾回收器详解重点垃圾回收器 算法的具体实现。分为新生代回收器和老年代回收器必须搭配使用。8.1 Serial 收集器串行单线程回收最简单、最古老回收时 STW 时间很长适用小内存、客户端、嵌入式设备搭配Serial Serial Old8.2 ParNew 收集器Serial 的多线程版本新生代专用是早期服务端唯一能和 CMS 配合的新生代回收器适用多 CPU 服务器8.3 Parallel Scavenge目标高吞吐量吞吐量 运行用户代码时间 / (运行用户代码时间 GC 时间)特点多线程并行自适应调节策略不追求低延迟追求CPU 利用率最高适用后台计算批处理日志分析不需要低延迟的场景8.4 Serial OldSerial 的老年代版本单线程标记 - 整理算法8.5 Parallel OldParallel Scavenge 的老年代版本多线程标记 - 整理高吞吐量生产常用组合Parallel Scavenge Parallel Old8.6 CMS 收集器低延迟神器全称Concurrent Mark Sweep目标最短 STW 停顿低延迟执行过程4 步初始标记STW只标记 GC Roots 直接关联对象极快并发标记和用户线程一起跑最慢但不卡顿重新标记STW修正并发标记期间变动的对象并发清除边跑业务边清除垃圾CMS 优点低延迟大部分过程并发用户几乎无感知CMS 缺点占用 CPU 资源产生内存碎片无法处理浮动垃圾并发失败会降级为 Serial Old导致长时间卡顿适用互联网项目、API 服务、支付、后台管理系统8.7 G1 收集器JDK 9 默认生产环境首选G1 Garbage First彻底抛弃分代结构把内存切成许多独立 Region。特点同时回收新生代 老年代可预测停顿时间优先回收垃圾最多的区域无内存碎片大内存6GB优势巨大优点低延迟高吞吐量稳定可控大堆最佳选择现在企业 90% 使用 G18.8 ZGC Shenandoah下一代超低延迟回收器停顿时间毫秒级以下几乎不 STW支持 TB 级内存云原生、大数据、实时系统未来主流九、Minor GC、Major GC、Full GC 区别9.1 Minor GC发生在新生代触发Eden 满频繁速度极快对系统影响小9.2 Major GC发生在老年代速度较慢频率低9.3 Full GC回收整个堆 元空间STW 最长最耗性能线上性能杀手触发 Full GC 的常见原因老年代空间不足元空间不足显式调用System.gc()CMS 并发失败Concurrent Mode Failure大对象无法分配十、GC 日志、命令、排查思路实战必备10.1 最常用 GC 查看命令plaintextjstat -gc 进程ID 1000 10重点看YGC新生代 GC 次数YGT新生代 GC 总时间FGCFull GC 次数FGCTFull GC 总时间健康标准FGC 越少越好单次 Full GC 时间 200msFull GC 后内存明显下降无泄漏10.2 线上 GC 问题表现接口频繁超时CPU 高但业务逻辑不复杂内存居高不下Full GC 频繁GC 后内存不下降 →内存泄漏十一、GC 总结最核心干货一句话记住GC 是 JVM 自动回收无用对象避免 OOM判断垃圾用可达性分析新生代对象存活率低 →复制算法老年代存活率高 →标记 - 清除 / 标记 - 整理STW是 GC 卡顿根源Serial/ParNew/Parallel 追求吞吐量CMS 追求低延迟G1 是目前生产环境绝对主流ZGC/Shenandoah 是未来超低延迟方向Full GC 是线上最大性能杀手必须尽量避免

相关文章:

彻底搞懂 Java 垃圾回收(GC)

在 Java 后端开发、面试、线上性能优化、OOM 排查中,GC(垃圾回收) 都是绕不开的核心基石。很多人只知道 GC 是自动回收内存,但到底怎么回收、什么时候回收、为什么会卡顿、不同回收器区别是什么,一知半解。这篇文章我用…...

基于混沌-高斯变异-麻雀搜索算法(CGSSA)优化BP神经网络(CGSSA-BP)的回归预测M...

基于混沌-高斯变异-麻雀搜索算法(CGSSA)优化BP神经网络(CGSSA-BP)的回归预测(含优化前后对比)MATLAB代码 代码注释清楚。 main为主程序,可以读取EXCEL数据。 很方便,容易上手。 &a…...

从ADB连接到权限修改:深入解析安卓APK安装的底层步骤

1. ADB连接:从物理连接到权限握手 很多人以为安卓APK安装就是双击文件那么简单,但当你需要调试系统级应用或修改预装应用时,就会发现事情没那么简单。我去年给某厂商定制系统应用时,光是为了让调试环境跑通就折腾了整整两天。下面…...

永磁同步电机PMSM的5+7次谐波注入与死区补偿策略:降低转矩脉动及电压补偿详解,附PPT、文...

永磁同步电机PMSM电机57次谐波注入,可以有效降低转矩脉动。 死区补偿后,有效降低转矩脉动。 电压补偿。 有ppt说明,文章和相应simulink模型。 描述真实,已更新,现在有两套模型。最近在调试永磁同步电机时发现个有意思的…...

探索多智能体系统中的事件触发控制代码

事件触发控制代码,每个代码有对应参考文献 1.多智能体中基于事件触发的协议 2.多智能体分布式系统的事件触发控制 3.基于观测器的非理想线性多智能体事件触发的跟踪一致性 4.非线性不确定扰动多智能体系统固定时间事件触发一致性控制 5.固定拓扑和切换多智能体分布式…...

短视频创作者的福音:Qwen3-ForcedAligner-0.6B毫秒级对齐,字幕制作效率翻倍

短视频创作者的福音:Qwen3-ForcedAligner-0.6B毫秒级对齐,字幕制作效率翻倍 1. 为什么短视频创作者需要精准字幕对齐? 在短视频内容爆炸式增长的今天,字幕已经成为提升观看体验的关键要素。数据显示,85%的观众会在静…...

锂电池温度检测Comsol仿真 软包锂电池表面温度变化仿真模拟,不同位置探针测温 #汽车级锂电池

锂电池温度检测Comsol仿真 软包锂电池表面温度变化仿真模拟,不同位置探针测温 #汽车级锂电池 Comsol仿真 最近在折腾汽车锂电池的温控仿真,发现软包电池的表面温度分布真是门玄学——同一个电池组里不同位置的温差能玩出花样。这次用COMSOL搞了个三维模…...

CH579 串口服务器 DTU 项目功能架构与实现解析

CH579 以太网转串口 串口服务器代码! 需要自己编程提升能力的非常值得参考的代码 几乎所有的编程思路编程技巧资源都涉及到了,代码简单易懂 ,注释清楚,本代码实现最串口服务器的功能,有电路图。CH579 串口服务器 DTU&a…...

CogVideoX-2b效果展示:看看这些由文字生成的精美短视频

CogVideoX-2b效果展示:看看这些由文字生成的精美短视频 1. 当文字开始流动:一次全新的视觉叙事体验 想象一下,你写下“一只戴着飞行员护目镜的柯基犬,在夕阳下的金色麦田里快乐奔跑”,然后点击一个按钮。两分钟后&am…...

避坑指南:VS2022中C#语言版本修改的正确姿势(含.NET Core版本查询技巧)

避坑指南:VS2022中C#语言版本修改的正确姿势(含.NET Core版本查询技巧) 当你在Visual Studio 2022中打开一个历史遗留项目时,是否遇到过这样的报错:"Feature xxx is not available in C# 7.3..."&#xff1f…...

西门子200smart PID算法源码探秘

西门子200smart PID算法源码,经过验证没问题 优点: 支持两路pwm输出与模拟量输出,可以用于恒温箱,一路控制加热一路控制制冷。 也可以用于恒压场合,一路控制加压阀一路控制泄压阀。 可以突破Pid向导8路限制最近在研究西门子200sma…...

Ubuntu 22.04 LTS下NVIDIA驱动安装避坑指南:如何用终端一键搞定(附常见错误解决)

Ubuntu 22.04 LTS下NVIDIA驱动安装避坑指南:如何用终端一键搞定(附常见错误解决) 在Linux系统上安装NVIDIA显卡驱动一直是让不少开发者头疼的问题。特别是对于Ubuntu 22.04 LTS用户来说,虽然系统本身对NVIDIA显卡的支持已经相当完…...

Claude Architect认证到底考什么?一个重度用户用半年实战逐项拆解

最近刷到一篇英文爆款:《I want to become a Claude architect (full course)》,756万浏览、5.6万收藏。作者把Anthropic官方的Claude Certified Architect考试大纲拆得底朝天。 我呢?用Claude Code写了整整大半年代码,从预测市场…...

别再死磕FTP了!手把手教你用SFTP连接Ubuntu虚拟机,FileZilla秒连成功

告别FTP连接困境:Ubuntu虚拟机SFTP配置全指南 每次在FileZilla里反复尝试FTP连接却总是失败?看着那些晦涩的错误提示却无从下手?作为开发者,我们经常需要在本地机器和Ubuntu虚拟机之间传输文件,而传统的FTP协议往往会成…...

Dell R730服务器部署Nvidia K80 GPU驱动与深度学习环境全攻略

1. 环境准备:从零开始的硬件与软件检查 在Dell R730服务器上部署Nvidia K80 GPU之前,我们需要像装修房子前检查地基一样做好准备工作。首先确认服务器已经正确安装了K80计算卡——这个双槽位的大家伙需要占用两个PCIe插槽,记得检查供电接口是…...

2026 AI财经落地实录:5个真实案例,告诉你具体怎么做才能见效

最近刷到不少讨论,说2026年AI在金融圈终于要“爆发”了。可我一查海外英文报告,发现好多大机构早就不是在“试水”,而是把AI直接塞进核心流程里,每天都在跑,省钱、省人力,还真金白银地降了风险。 你以为AI…...

U8g2自定义中文字库实战:从零构建Arduino OLED专属字体

1. 为什么需要自定义U8g2中文字库 在嵌入式开发中,我们经常会遇到需要在OLED屏幕上显示中文的需求。使用U8g2库自带的完整中文字库虽然方便,但对于存储空间有限的开发板(如Arduino UNO)来说,这可能会带来严重的问题。 …...

6.4 日志到底怎么写才有用?排障效率提升的底层方法

第6章 第4节:日志到底怎么写才有用?排障效率提升的底层方法 章节主题:安全测试与工程质量 关键词:AI协作、产品交付、工程化、可持续迭代 一、开场:为什么这件事值得你现在就做 很多读者问过同一个问题:日志到底怎么写才有用?排障效率提升的底层方法。 在大量项目复盘…...

05_Priority Queues 优先队列

title: 05_Priority Queues 优先队列 categories: 02_Silver tags: 优先队列堆Priority QueueHeap Priority Queues 优先队列 简介 优先队列(Priority Queue 或 Heap)支持以下操作: 插入元素删除最高优先级元素获取最高优先级元素 以上操…...

等保三级下主流厂商网络设备安全配置实战指南

1. 等保三级网络设备安全配置的核心要求 等保三级作为国内网络安全等级保护的重要标准,对网络设备的安全配置提出了明确要求。在实际项目中,我经常遇到工程师对等保要求理解不到位的情况,导致设备配置反复修改。这里我结合多年经验&#xff0…...

6.3 能跑不等于能交付:测试分层与回归方案

第6章 第3节:能跑不等于能交付:测试分层与回归方案 章节主题:安全测试与工程质量 关键词:AI协作、产品交付、工程化、可持续迭代 一、开场:为什么这件事值得你现在就做 很多读者问过同一个问题:能跑不等于能交付:测试分层与回归方案。 在大量项目复盘中可以看到,真正…...

ComfyUI文生图工作流参数调优实战:从新手到进阶的5个关键技巧

ComfyUI文生图工作流参数调优实战:从新手到进阶的5个关键技巧 当你已经能够用ComfyUI生成基本图像后,是否遇到过这些困扰:明明用了精心设计的提示词,结果却总差强人意?生成的人物面部细节模糊得像打了马赛克&#xff1…...

GenICam GenTL 标准 ver1.5(2)GenTL传输层:连接相机与应用的桥梁

1. GenTL传输层:机器视觉的"数据高速公路" 想象一下你正在建设一个智能工厂,需要把20台不同品牌的工业相机接入同一个检测系统。有的相机用GigE网线传输数据,有的用USB3.0接口,还有的使用Camera Link HS高速接口——这就…...

Avalonia 开发环境配置全攻略:从零搭建到高效开发

1. Avalonia开发环境搭建入门指南 第一次接触Avalonia的开发者可能会被各种配置步骤搞得晕头转向。作为一个跨平台的.NET UI框架,Avalonia确实需要一些前期准备工作才能开始愉快的编码之旅。不过别担心,跟着我的步骤走,保证你能在半小时内搞定…...

手把手教你用DiskGenius给瘦客户机分区(WinPE环境实操指南)

瘦客户机系统部署实战:WinPE环境下DiskGenius分区与系统安装全解析 瘦客户机作为企业级精简计算设备,其系统部署与传统PC存在显著差异。许多IT运维人员在初次接触这类设备时,往往会被其特殊的硬件架构和系统要求所困扰。本文将深入探讨如何在…...

ArcGIS 10.2安装与汉化全流程指南:从零开始搭建专业地理信息平台

1. ArcGIS 10.2入门:为什么选择这个经典版本? ArcGIS 10.2作为地理信息系统领域的里程碑版本,至今仍是许多企业和科研机构的首选。我在实际项目中发现,这个版本在稳定性和功能完整性上达到了很好的平衡。相比新版,它对…...

WGCNA分析实战指南:从基因模块挖掘到关键基因鉴定

1. WGCNA分析入门:为什么你需要掌握这个工具 第一次接触WGCNA这个词的时候,我也是一头雾水。直到在分析一批植物抗旱基因表达数据时,传统方法怎么也找不出关键调控基因,导师建议我试试WGCNA,结果让我大吃一惊——它不仅…...

深入解析CMake路径变量:CMAKE_CURRENT_SOURCE_DIR与CMAKE_CURRENT_LIST_DIR的实战对比

1. 初识CMake路径变量:从项目结构说起 第一次接触CMake时,很多人会被各种路径变量搞得晕头转向。就拿最常见的CMAKE_CURRENT_SOURCE_DIR和CMAKE_CURRENT_LIST_DIR来说,它们看起来都能获取当前路径,但在实际项目中表现却大不相同。…...

A星算法(A*)从入门到精通:手把手教你实现路径规划代码

1. 什么是A星算法? 第一次听说A星算法时,我也是一头雾水。直到把它想象成现实生活中的导航系统,才恍然大悟。简单来说,A星算法就像是一个聪明的向导,能在复杂的地图中帮你找到从起点到终点的最佳路线。 这个算法最早出…...

FlowState Lab大模型部署实战:基于Python的快速环境搭建与模型调用

FlowState Lab大模型部署实战:基于Python的快速环境搭建与模型调用 1. 开篇:为什么选择FlowState Lab? 如果你正在寻找一个既强大又容易上手的大模型开发环境,FlowState Lab绝对值得一试。作为一个专为AI开发者设计的开源框架&a…...