Java如何判断对象已死亡?

引用计数法:

程序给对象添加一个引用计数器,每有一个变量引用它时计数器加1,当引用断开时计数器减一。当计数器为0时,代表没有任何变量引用他,该对象就死亡状态,JVM需要对此类对象进行回收。此计数法无法回收具有循环引用的对象,由于对象相互引用,导致各自计数器都不为0,导致JVM无法回收。

可达性分析法:

程序由GC Roots作为起点,从这些节点开始向下搜索,搜索走过的路径称为引用链,当一个对象到GC Roots没有任何引用链时,证明此对向是不可用的。 Java中,作为GC Roots的对象有一下几种:

  1. 虚拟机栈(栈帧中的本地变量表)中的引用对象
  2. 方法区中类静态属性引用的对象
  3. 方法区中常量引用的对象
  4. 本地方法栈(native方法)中引用的对象
/**
 * 可作为GC Roots的对象
 * 1、虚拟机栈(栈帧中的本地变量表)中引用的对象
 * 2、方法区中类静态属性引用的对象
 * 3、方法区中常量引用的对象
 * 4、本地方法栈中JNI(Native方法)引用的对象(和虚拟机栈类似,一个是虚拟机层面的调用,一个是本地层面的调用)
 **/
public class Demo1 {
    /**
     * 1、虚拟机栈(栈帧中的本地变量表)中引用的对象
     * 此时的 s,即为 GC Root,当s置空时,parameter 对象也断掉了与 GC Root 的引用链,将被回收
     */
    public static void testGC1() {
        StackLocalParameter s = new StackLocalParameter("parameter");
        s = null;
    }

    /**
     * 2、方法区中类静态属性引用的对象
     * s 为 GC Root,s 置为 null,经过 GC 后,s 所指向的 staticProperties 对象由于无法与 GC Root 建立关系被回收。
     * 而 m 作为类的静态属性,也属于 GC Root,parameter 对象依然与 GC root 建立着连接,所以此时 parameter 对象并不会被回收。
     */
    public static void testGC2() {
        MethodAreaStaicProperties s = new MethodAreaStaicProperties("staticProperties");
        s.m = new MethodAreaStaicProperties("parameter");
        s = null;
    }

    /**
     * 3、方法区中常量引用的对象
     * m 即为方法区中的常量引用,也为 GC Root,s 置为 null 后,final 对象也不会因没有与 GC Root 建立联系而被回收
     */
    public static void testGC3() {
        MethodAreaConstant s = new MethodAreaConstant("parameter");
        s = null;
    }
}

/**
 * 1、虚拟机栈
 */
class StackLocalParameter {
    public StackLocalParameter(String name) {
    }
}

/**
 * 2、方法区静态属性
 */
class MethodAreaStaicProperties {
    public static MethodAreaStaicProperties m;

    public MethodAreaStaicProperties(String name) {
    }
}

/**
 * 3、方法区中常量引用的对象
 */
class MethodAreaConstant {
    public static final MethodAreaConstant m = new MethodAreaConstant("final");

    public MethodAreaConstant(String name) {
    }
}

猜你喜欢

转载自blog.csdn.net/Ahuuua/article/details/124732230
今日推荐