1. 垃圾回收概述
1.1 什么是垃圾?
技术定义:
在JVM语境中,"垃圾"指失去所有引用的堆内存对象。判定标准:
关键特征:
- 不可达对象(通过GC Roots不可达)
- 循环引用孤岛(彼此引用但整体不可达)
- 特殊引用(软/弱/虚引用)的次级对象
1.2 为什么需要GC?
三大核心原因:
- 内存安全:防止野指针和非法内存访问
- 资源管理:自动回收不再使用的对象
- 性能优化:通过内存整理提升访问效率
1.3 早期垃圾回收
技术演进路线:
1.4 Java垃圾回收机制
核心架构:
关键特性:
- 自动内存管理:开发人员无需显式释放
- 分代假设:弱分代假说/强分代假说
- STW机制:安全点(Safepoint)控制
2. 垃圾回收相关算法
2.1 标记阶段:引用计数算法
实现原理:
致命缺陷演示:
循环引用问题:即使对象间互相引用,但因外部引用断开,实际已成为垃圾却无法被识别
2.2 标记阶段:可达性分析算法
核心原理:
GC Roots类型:
- 虚拟机栈中的局部变量
- 方法区静态属性引用
- 方法区常量引用
- 本地方法栈JNI引用
- 同步锁持有对象
2.3 对象的finalization机制
生存判定流程:
重要特性:<font style="background-color:rgb(252, 252, 252);">finalize()</font>方法只会被JVM调用一次,且不保证执行顺序
2.4 MAT与JProfiler的GC Roots溯源
实战操作流程:
JProfiler溯源示例:
2.5 清除阶段:标记-清除算法
执行过程详解:
- 标记阶段:通过可达性分析标记所有存活对象
- 清除阶段:线性遍历堆内存,回收未被标记的对象块
缺陷分析:
技术痛点:
- 两次全堆扫描(标记和清除)效率低
- 清除后内存空间不连续,导致分配大对象时触发Full GC
2.6 清除阶段:复制算法
核心机制:
- 内存划分:将可用内存分为两个等大的From和To区域
- 存活对象迁移:将From区存活对象复制到To区,并保持内存紧凑
优劣对比表:
| 优势 | 劣势 |
|---|---|
| 无内存碎片 | 内存利用率仅50% |
| 高速分配(指针碰撞) | 对象存活率高时效率骤降 |
| 适合年轻代 | 需要额外空间处理引用更新 |
实战应用:
2.7 清除阶段:标记-压缩算法
指针碰撞原理:
适用场景:
- 老年代垃圾回收(配合CMS或G1使用)
- 需要长期存活的大对象管理
性能影响:
2.8 算法对比总结
2.9 分代收集算法
内存代际划分:
回收策略矩阵:
| 代际 | 算法 | 触发条件 | 耗时 |
|---|---|---|---|
| 年轻代 | 复制算法 | Eden满 | 毫秒级 |
| 老年代 | 标记-压缩 | 空间不足 | 秒级 |
对象晋升流程:
2.10 增量收集与分区算法
增量收集原理:
分区算法优势:
3. 垃圾回收常见问题与解决方案
3.1 内存泄漏问题
典型案例:
- 线程局部变量泄漏:ThreadLocal使用后未remove
- 缓存失控增长:Guava Cache未设置过期策略
- JNI引用未释放:本地方法分配的内存未回收
3.2 GC性能调优
调优决策树:
参数配置矩阵:
| 场景 | 推荐参数 | 作用域 |
|---|---|---|
| Web应用 | -XX:+UseG1GC | 全堆 |
| 大数据计算 | -XX:+UseParallelGC | 年轻代 |
| 低延迟交易 | -XX:+UseZGC | JDK11+ |
3.3 Full GC频繁触发
诊断流程图:
常见诱因:
- 老年代空间分配担保失败
- 元空间/metadata区溢出
- System.gc()主动调用
4. 高频面试问题与解答
Q1:引用计数算法与可达性分析的本质区别?
答案要点:
Q2:对象自救的可行性及限制?
技术解析:
关键限制:
- finalize()执行顺序不确定
- 自救仅能执行一次
- 不推荐生产环境使用
Q3:标记-清除算法导致的内存碎片如何影响系统?
影响路径:
解决方案对比:
| 方案 | 原理 | 副作用 |
|---|---|---|
| 标记-压缩 | 内存滑动整理 | STW时间增加 |
| 空闲列表 | 维护可用块记录 | 分配效率下降 |
Q4:G1收集器如何实现可预测停顿?
核心机制:
技术亮点:
- 将堆划分为2048个Region(默认)
- 基于回收效益的优先级排序
- 使用Remembered Set处理跨代引用
Q5:如何排查OOM问题?
实战检查清单:
关键命令:
# 生成dump文件
jmap -dump:format=b,file=heap.hprof <pid>
# 查看对象直方图
jmap -histo <pid>
Q6:ZGC的核心创新点?
技术突破:
适用场景:
- 要求低延迟的金融交易系统
- 大内存云原生应用
- JDK15+生产环境