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

堆的分代与垃圾回收

一句话

JVM 根据"大部分对象朝生夕死"的规律,把堆分成了​新生代和老年代​,不同代用不同的回收策略。理解对象在堆里的流转过程,就掌握了 GC 的核心。


为什么堆要分代?

因为​大部分对象活不久​:

你的项目里: 查一次列表 → new 一堆 User 对象 → 用完没人引用了 下一次请求 → 又 new 一堆 → 又没人引用了 → 90% 的对象活不过几毫秒

如果不分代,每次回收都要扫描整个堆,效率极低。分代后:

  • 新生代​:空间小,回收快(Minor GC,毫秒级)
  • 老年代​:放长命对象,回收频率低(Full GC,秒级)

堆的分区结构

堆(Heap) │ ├── 新生代(Young Generation) │ ├── Eden(E 甸园) ← 占 80% │ └── Survivor 区 │ ├── S0(From) ← 占 10% │ └── S1(To) ← 占 10% │ └── 比例:Eden : S0 : S1 = 8 : 1 : 1 │ └── 老年代(Old Generation)

默认新生代和老年代比例约 ​1 : 2​(可调)。


对象完整流转过程

new User() → 放进 Eden 区 │ ├── Eden 快满了 → Minor GC │ │ │ ├── 没人引用的对象 → 回收 │ └── 还有人引用的对象 → 移到 S0(年龄=1) │ ├── Eden 再次满了 → Minor GC │ │ │ ├── Eden 存活对象 + S0 存活对象 → 移到 S1(年龄+1) │ └── S0 清空 │ ├── 反复如此:Eden + 正在使用的 Survivor → 另一个 Survivor │ S0 和 S1 始终保持一个为空 │ ├── 对象年龄达到阈值(默认 15)→ 升到老年代 │ ├── 或者:Survivor 区装不下了 → 动态年龄判定 → 年龄大的提前升老年代 │ └── 老年代满了 → Full GC │ ├── 扫描整个堆(新生代 + 老年代 + 元空间) └── STW(Stop The World):所有线程暂停

Minor GC vs Full GC

对比项Minor GC(Young GC)Full GC
回收范围新生代(Eden + Survivor)整个堆 + 元空间
触发条件Eden 区满了老年代满了 / 元空间满了 / 主动调用 System.gc()
STW 时长几毫秒(用户基本无感知)几秒甚至几十秒(系统卡顿)
频率频繁少(但每次都很重)

注意​:Minor GC 也会 ​**STW(Stop The World)**​——暂停所有线程。但因为只扫新生代,范围小,所以很快。

Full GC 是 JVM 调优的主要目标​——要尽量降低 Full GC 的频率和时长。


GC Roots 和可达性分析

JVM 判断对象"该不该回收",用的是​可达性分析​:

从 GC Roots(肯定活着的起点)出发 能走到的对象 → 活着 ❌ 不回收 走不到的对象 → 不可达 → 回收 ✅ GC Roots 包括: ├── 栈帧中的局部变量引用(方法里正在用的对象) ├── 静态变量引用(static 修饰的) └── 活跃线程

打个比方​:

一栋楼(堆)里有 1000 个房间 你不需要查每个房间有没有人 你只需要从 3 个值班室(GC Roots)出发: 前台有人的 → 她旁边的房间也有人 紧急指示灯亮着 → 它的线路都有人维护 监控室有人 → 监控覆盖的区域都是活的 能走到的房间 → 有人 → 不回收 走不到的房间 → 没人 → 回收

面试问答

Full GC 频繁怎么排查?

① 看监控:老年代占用率持续上升 ② 加 JVM 参数:-XX:+HeapDumpOnOutOfMemoryError ③ OOM 时拿到 hprof 堆转储文件 ④ MAT 分析:看占内存最大的对象 ⑤ 定位代码:顺着线程栈找到问题 SQL/代码 常见原因: - SQL 没有分页,一次查太多数据 - 定时任务频率太高 - 内存泄漏(对象只增不减)

一个对象一定会熬到 15 岁才进老年代吗?

不一定。还有​动态年龄判定​:如果 Survivor 区装不下了,会把年龄最大的那批对象提前升到老年代,不一定要等到 15 岁。


参考来源

  • 《深入理解 Java 虚拟机》第 3 章
  • JDK8 G1 GC 文档
http://www.jsqmd.com/news/1098885/

相关文章:

  • 终极Windows窗口强制调整工具:轻松解决顽固窗口大小问题
  • Web漏洞扫描工具实战指南:从选型配置到自动化集成
  • Python之yandex-annlib包语法、参数和实际应用案例
  • 【JAVA毕设源码分享】基于springboot二手滑板交易系统的设计与实现(程序+文档+代码讲解+一条龙定制)
  • 类?.调用方法()这种写法的解释
  • 部署上线 GitHub+Vercel+CloudFlare
  • 数字校园SQL注入防御:从原理到实战的纵深检测与动态响应体系
  • 数据分析师成长路径:从思维到工具,构建解决实际问题的核心能力
  • Windows系统文件hidserv.dll丢失找不到问题解决
  • 大厂必考 Binder 底层:in/out/inout/oneway 关键字、IPC 性能差异满分解析
  • DART:采样两份草稿估计思考预算,节省 67% token 效果还更好
  • ai-image-gen-mcp MCP 服务说明文档
  • 数据安全检查,这3个API盲区最容易被问穿
  • Windows 11 点击“电源和电池”设置直接闪退?罪魁祸首竟然是Sensor Service!
  • 机器学习与模式识别 第一章 机器学习导论 考点压缩
  • 吃透Spring事务 :核心原理,传播机制,隔离级别,使用场景
  • 自动化测试框架选型与Robot Framework环境搭建实战指南
  • Windows Defender深度控制架构设计与系统级安全策略管理实现
  • 基于改进YOLOv8与无人机航拍的电动自行车违规行为智能检测系统实战
  • E-Hentai下载器完整指南:如何轻松批量下载并打包图片资源
  • 如何快速部署Python自动化脚本:京东商品预约下单的完整解决方案
  • 第一次学 volatile 关键字,我看了三遍才搞懂它到底在干嘛
  • 如何免费使用Outfit字体:9种字重打造专业品牌设计的完整指南
  • 别再傻傻手写了!Python一行代码判断是不是数字,爽到飞起
  • Ansible自动化运维实战:从入门到精通,轻松管理服务器集群
  • JVM 运行时数据区 —— 5 大块内存
  • C++ Primer Plus 重读精讲 _ 指针进阶全集:三类const指针辨析、指针数组数组指针硬核区分、指针地址传参、工控函数双向改参实战
  • VMware虚拟机组网通信全链路解析(ESXi 7.0+vSphere 8.0实测验证)
  • 面向对象——多态
  • Focus架构:视觉语言模型的高效加速方案