2024-10-26
版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。 本文链接:https://www.skjava.com/mianshi/baodian/detail/1121523237

回答

GC Roots 是可达性分析算法的起点,那有哪些对象可以作为 GC Roots 呢?

  • 栈帧中的局部变量和参数

这类对象属于上下文中的对象。当线程在执行方法时,它会将方法打包成一个栈帧压入到栈中去执行,方法里用到的局部变量会存放到栈帧的本地变量表中。只要方法还在执行,还没有出栈,就以为这些本地变量表中的对象还会被访问,GC 就不能回收。所以,这类对象可以作为 GC Roots

例如,一个方法中定义的局部变量的引用,如果该方法正在执行中,那么这些引用的对象会被认为是存活的:

public void test() {
    Object obj = new Object(); // 局部变量 obj 引用的对象可能作为 GC Roots
}
  • 方法区常量池引用的对象

常量池中的对象是全局的,它在整个应用程序运行期间是有效的,所以,作为 GC Roots 也不过分。

例如,字符串常量 "skjava.com" 存在于常量池中,而常量池中的引用也会被视为 GC Roots。

String str = "skjava.com"; // 常量池中的字符串常量
  • 方法区静态变量引用的对象

静态变量属于类级别的变量,它们在整个应用程序生命周期内都存在。当类被加载时,静态属性被初始化并持有对象的引用。同时,Class 对象本身是非常难被回收的,只要 Class 对象不回收,静态属性就不能被回收。

public class Test {
    public static Object staticObj = new Object(); // staticObj 引用的对象可能作为 GC Roots
}
  • JNI本地方法栈中引用的对象

Java 可以通过 JNI(Java Native Interface)调用本地代码,而 JNI 中的本地代码可以持有对 Java 对象的引用,这些引用是不能被回收的。所以,也会被视为 GC Roots。

  • Java 虚拟机内部的引用

JVM 内部的一些特殊数据结构也可能包含对对象的引用,例如,某些 JVM 内部的管理结构、类加载器相关的对象等。这些对象也会作为 GC Roots。

  • 被同步锁持有的对象

使用 synchronized 关键字进行同步的对象,会被视为 GC Roots。这些对象在同步块或者方法执行期间会被 JVM 保留,防止被垃圾回收。回收了,锁咋搞?

synchronized (lockObject) {
    // lockObject 持有锁,被视为 GC Roots
}
阅读全文