登录
首页 >  文章 >  java教程

JavaGC对象存活判定方法详解

时间:2025-12-08 14:15:48 428浏览 收藏

推广推荐
免费电影APP ➜
支持 PC / 移动端,安全直达

学习文章要努力,但是不要急!今天的这篇文章《Java GC如何判定对象存活?可达性算法详解》将会介绍到等等知识点,如果你想深入学习文章,可以关注我!我会持续更新相关文章的,希望对大家都能有所帮助!

Java GC采用可达性分析算法判定对象存活,从GC Roots(如栈中局部变量、静态属性、常量、JNI引用等)出发搜索引用链,不可达对象即被回收;循环引用不影响回收,软/弱/虚引用按策略处理。

Java的GC如何判定对象存活_Java可达性算法与标记阶段解析

Java的垃圾回收(GC)并不依赖引用计数,而是采用可达性分析算法来判定对象是否存活。核心逻辑很简单:从一组称为“GC Roots”的对象出发,沿着引用链向下搜索,能被访问到的对象视为“存活”,其余则标记为可回收。

哪些对象可以作为GC Roots?

GC Roots不是随便选的,必须是JVM能明确保证生命周期、不会被回收的起点。常见包括:

  • 虚拟机栈(栈帧中的局部变量表)中引用的对象
  • 方法区中类静态属性引用的对象
  • 方法区中常量引用的对象(如字符串常量池里的String实例)
  • 本地方法栈中JNI(即Native方法)引用的对象
  • 正在被同步锁(synchronized)持有的对象(JDK 9+部分实现中考虑)

可达性分析 ≠ 引用链不断开就一定存活

即使某个对象还被引用着,也不代表它一定“真正有用”。比如软引用(SoftReference)、弱引用(WeakReference)、虚引用(PhantomReference)所指向的对象,在GC时会按不同策略处理:

  • 强引用:最常见,如 Object obj = new Object(),只要强引用存在,GC就不会回收
  • 软引用:内存不足时才回收,适合做缓存(如SoftReference
  • 弱引用:每次GC都会尝试回收,常用于避免内存泄漏(如ThreadLocal中的key)
  • 虚引用:无法通过它获取对象,唯一作用是在对象被回收前收到系统通知(配合ReferenceQueue使用)

标记阶段做了什么?

在可达性分析完成后,GC进入标记阶段——把所有从GC Roots可达的对象打上“存活”标记。不同收集器实现略有差异,但本质一致:

  • 初始标记(Initial Mark):暂停用户线程(STW),快速扫描GC Roots直接关联的对象(如栈中引用、静态字段等)
  • 并发标记(Concurrent Mark):与用户线程并发执行,遍历整个对象图,标记所有可达对象(CMS、G1、ZGC都支持)
  • 重新标记(Remark):再次STW,修正并发期间因用户线程修改引用关系导致的漏标(比如A→B断开、C→B建立)

注意:标记本身不清理内存,只是为后续清除或整理阶段提供依据。

一个容易忽略的关键点:循环引用不会阻止回收

Java不靠引用计数,所以两个对象互相引用(A → B,B → A),但都没被GC Roots连通,它们依然会被回收。这是和Python、Objective-C等语言的重要区别。

比如:

class Node { Node next; }
Node a = new Node();
Node b = new Node();
a.next = b;
b.next = a;
a = null; b = null;

此时a、b已不可达,下次GC就会回收它们。

基本上就这些。理解GC Roots的范围、引用类型的语义、以及标记阶段的实际行为,比死记“哪些算法”更有助于排查内存问题。

以上就是《JavaGC对象存活判定方法详解》的详细内容,更多关于的资料请关注golang学习网公众号!

相关阅读
更多>
最新阅读
更多>
课程推荐
更多>