回答
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
}