Java垃圾回收本质是自动管理堆内存,通过可达性分析(以GC Roots为起点)判定对象存活,按分代理论将堆分为新生代(Eden+S0/S1)和老年代,分别采用复制、标记-清除/整理算法回收;Minor GC在Eden满时触发,Full GC在老年代不足等情况下触发。
Java 中的垃圾回收(GC)本质是自动管理堆内存,识别并回收不再被引用的对象,释放空间供新对象使用。核心在于判断“哪些对象可被回收”,以及“何时、如何回收”。
Java 不采用引用计数法(因无法解决循环引用),而是基于可达性分析(Reachability Analysis):以一组称为“GC Roots”的对象为起点,向下搜索引用链;所有能被这条链触及的对象视为“存活”,其余即为可回收对象。
常见的 GC Roots 包括:
HotSpot 虚拟机将堆分为新生代(Young Gen)和老年代(Old Gen),依据是“绝大多数对象朝生夕灭”的经验规律。
新生代进一
步划分为 Eden 区 + 两个 Survivor 区(S0、S1)。对象优先分配在 Eden;一次 Minor GC 后,存活对象进入 Survivor;经历多次 Minor GC 仍存活(默认 15 次,由 -XX:MaxTenuringThreshold 控制),则晋升至老年代。
这种划分让 GC 更高效:新生代用复制算法(快、无碎片),老年代用标记-清除或标记-整理(空间利用率高)。
不同代采用不同算法:
主流收集器如 G1、ZGC、Shenandoah 已不再严格按分代,但依然依赖可达性分析 + 增量/并发标记等机制来降低 STW 时间。
GC 不是定时执行,而是由内存压力驱动:
可通过 -XX:+PrintGCDetails -Xloggc:gc.log 查看 GC 日志,分析频率、停顿、回收效果。
基本上就这些。GC 的设计是在吞吐量、延迟、内存占用之间做权衡,理解原理有助于合理调参、定位 OOM 或频繁 GC 问题,而不是盲目调大堆或禁用 System.gc。