Java GC采用可达性分析算法判定对象存活,从GC Roots(如栈中局部变量、静态属性、常量、JNI引用等)出发搜索引用链,不可达对象即被回收;循环引用不影响回收,软/弱/虚引用按策略处理。
Java的垃圾回收(GC)并不依赖引用计数,而是采用可达性分析算法来判定对象是否存活。核心逻辑很简单:从一组称为“GC Roots”的对象出发,沿着引用链向下搜索,能被访问到的对象视为“存活”,其余则标记为可回收。
GC Roots不是随便选的,必须是JVM能明确保证生命周期、不会被回收的起点。常见包括:
即使某个对象还被引用着,也不代表它一定“真正有用”。比如软引用(SoftReference)、弱引用(WeakReference)、虚引用(PhantomReference)所指向的对象,在GC时会按不同策略处理:
Object obj = new Object(),只要强引用存在,GC就不会回收SoftReference)
引用:无法通过它获取对象,唯一作用是在对象被回收前收到系统通知(配合ReferenceQueue使用)在可达性分析完成后,GC进入标记阶段——把所有从GC Roots可达的对象打上“存活”标记。不同收集器实现略有差异,但本质一致:
注意:标记本身不清理内存,只是为后续清除或整理阶段提供依据。
Java不靠引用计数,所以两个对象互相引用(A → B,B → A),但都没被GC Roots连通,它们依然会被回收。这是和Python、Objective-C等语言的重要区别。
比如:
class Node { Node next; }此时a、b已不可达,下次GC就会回收它们。
基本上就这些。理解GC Roots的范围、引用类型的语义、以及标记阶段的实际行为,比死记“哪些算法”更有助于排查内存问题。