GC root原理
通过对枚举GCroot对象做引用可达性分析,即从GC root对象开始,向下搜索,形成的路径称之为 引用链。如果一个对象到GC roots对象没有任何引用,没有形成引用链,那么该对象等待GC回收。
可以作为GC Roots的对象
1、虚拟机栈(javaStack)(栈帧中的局部变量区,也叫做局部变量表)中引用的对象。
如下代码所示,a 是栈帧中的本地变量,当 a = null 时,由于此时 a 充当了 GC Root 的作用,a 与原来指向的实例 new Test() 断开了连接,因此对象会被回收。
publicclass Test {
public static void main(String[] args) {
Test a = new Test();
a = null;
}
}
2、方法区中的类静态属性引用的对象。
如下代码所示,我们给 s 赋值了变量的引用,s 在此时是类静态属性引用,充当了 GC Root 的作用,它指向的对象存活
public class Test {
public static Test s = new Test();
}
3、方法区中常量引用的对象。
如下代码所示,常量 s 指向的对象存活
public class Test {
public static final Test s = new Test();
}
4、本地方法栈中JNI(Native方法)引用的对象。
本地方法就是一个 java 调用非 java 代码的接口,该方法并非 Java 实现的,可能由 C 或 Python等其他语言实现的,Java 通过 JNI 来调用本地方法
当调用 Java 方法时,虚拟机会创建一个栈桢并压入 Java 栈,而当它调用的是本地方法时,虚拟机会保持 Java 栈不变,不会在 Java 栈祯中压入新的祯,虚拟机只是简单地动态连接并直接调用指定的本地方法。
如下代码所示,当 java 调用以上本地方法时,jc 会被本地方法栈压入栈中, jc 就是我们说的本地方法栈中 JNI的对象引用,因此只会在此本地方法执行完成后才会被释放。
JNIEXPORT void JNICALL Java_com_newStringNative(JNIEnv *env, jobject instance,jstring jmsg) {
...
// 缓存String的class
jclass jc = (*env)->FindClass(env, STRING_PATH);
}