JVM中的GC是如何判断对象可回收

Java作为一种近几十年兴起的编程语言,由于其提供了完整的用于软件开发和跨平台部署的支持环境,即实现了“Write Once, Run Anywhere ”梦想,因此受到了越来越多互联网公司的青睐,使用Java语言编写程序以及从事其编程工作的人员越来越多,作为一名合格的中高级Java工程师以及系统架构师,深入理解JVM虚拟机的运行机理成为必不可少的一部分,而JVM的垃圾回收机制又是其重要的一环,下面对GC回收前如何判断对象可回收进行简单的解析。

目录

  1. 如何判断对象可回收
  2. GC怎样进行回收

一、如何判断对象已死

针对如何判断对象可回收很多人给出如下答案:给对象添加一个引用计数器,当有一个地方引用它时,计数器的值就加1,;当引用失效时,计数器的值就减1,如果一个对象的引用计数器为0则表示这个对象已不可能再次被引用,GC就可进行回收,这种算法在实际中是存在缺陷的,如下代码

public class MyObject{
    public MyObject myObject;
    //当创建实例时,主要有此属性占据内存,如果清理下面创建的实例对象
    //则肯定会显示此次回收了多少内存空间,回收内存空间的大小由此属性决定
    private byte[] bytes = new byte[2*1024*1024]
    public static void main(String[] args){
        MyObject mObj1 = new MyObject();
        MyObject mObj2 = new MyObject();
        //mObj1和mObj2相互引用
        mobj1.myObject = mObj2;
        mobj2.myObject = mObj1;
        /**
        * mObj1和mObj2两个变量都赋值为空,但是在堆(Heap)内存中却创建
        * 的类MyObject的两个实例,并且由于存在相互引用,每个实例对象的引
        * 用计数器应该是都为1的,按照计数器清除原理本不应该被清除,但是按
        * 照正常的逻辑,这个两个对象由于不可能再次被引用到,应该被清除掉的,
        * 在现实的程序运行中,也的确被清除了,这说明了JVM采用的并不是这
        * 种方式.
        */
        mObj1 = null;
        mObj2 = null;
        //调用GC,怎样查看对象是否清楚请读者独自查看相关资料
        System.gc();
    }
}

通过对上面代码的分析,计数器算法的确存在缺陷,并且JVM本身采用的也不是此种算法。实际上,Java语言为了避免上述算法存在的缺陷,其通过一系列名为“GC Roots”的对象作为起始点,从这个节点开始向下搜索,从而形成许多引用路径链,当一个对象没有任何一条引用链可以抵达起始点时,则说明此对象已经不会再被引用,则可进行回收,在Java语言中,可作为GC Roots对象主要包括以下几种:

  • 虚拟机栈(栈帧中的本地变量表)中的引用对象
  • 方法区中的类静态属性引用的对象
  • 方法区中的常量引用的对象
  • 本地方法栈中JNI(即一般说的Native方法)引用的对象。

二、GC怎样进行回收

在GC正式执行回收前,首先要提到实例对象的finalize()方法,实例对象会有一个计数器记录这个方法的调用次数,即未调用过计数器属性值为0,调用过计数器属性值为1。GC在判断过哪些对象没有被引用后,会进一步判断finalize()方法是否被调用过,如果已经被调用过直接进行回收,如果没有被调用过则会将这些对象放入一个名为F-Queue的队列之中,然后再对队列中的实例执行finalize()方法,在finalize()方法内部可完成实例的重新被引用(如下代码),然后GC会再次对F-Queue队列中实例对象进行整理,将没有被引用的对象进行回收,将重新被引用的对象从队列F-Queue删除,至此一次GC垃圾回收完成,下次GC的垃圾回收仍会按照这种执行流程执行。

public class MyObject{
    //用来使实例重新获得引用
    public static MyObject myObject = null;
    //用来判断实例仍然活着
    public void isAlive(){
        System.out.println("myObject实例仍然活着......");
    }
    public void finalize() throws Throwable(){
        super.finalize();
        System.out.println("finalize()方法执行了......");
        //将实例重新获得引用
        myObject = this;
    }
    public static void main(String[] args) throws Trowable{
        myObject = new MyObject();
        /**
        * 实例第一次失去引用,GC会执行实例的finalize()方法
        * 在执行finalize()方法的过程中,实例重新获得引用,
        * 同时实例将记录finalize()方法执行次数的属性值赋为1
        */
        myObject = null;
        System.gc();//输出finalize()方法执行了......
        //线程睡眠5秒,以便让垃圾回收器执行
        Thread.sleep(5000);
        if (myObject != null){
            myObject.isAlive();//输出myObject实例仍然活着......
        } else {
            System.out.println(myObject实例已经死了......);
        }

        //实例第二次失去引用
        myObject = null;
        /**
        * 由于实例finalize()方法执行计数器已经变为1,此次不再执行
        * finalize()方法,直接进行回收
        */
        System.gc();
        //线程睡眠5秒,以便让垃圾回收器执行
        Thread.sleep(5000);
        if (myObject != null){
            myObject.isAlive();
        } else {
            System.out.println(myObject实例已经死了......);//执行
        }

    }
    
}

以上分析是GC垃圾回收的执行机理,供大家交流学习。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Java虚拟机判断一个对象是否可以回收的方法是通过垃圾收集器对内存空间的扫描。如果发现一个对象没有被任何指针或者引用所指向,也没有被任何活动线程所持有,那么这个对象就可以被回收。Java虚拟机还提供了一种判断方式,叫做“可达性分析”。可达性分析的基本思想就是通过一系列称为“GC Roots”的对象作为起始点,从这些节点开始向下搜索,搜索所走过的路径称为“引用链”,当一个对象GC Roots没有任何引用链相连时,则证明此对象是不可用的。 ### 回答2: JVM判断一个对象是否可以回收主要依靠垃圾回收算法和GC Root对象。以下是判断对象是否可以回收的几个条件: 1. 引用计数法(Reference Counting):每当一个对象被引用时,引用计数加1,引用失效时计数减1。当计数器为0时,说明该对象没有任何引用,可以判断可回收对象。然而,该算法难以解决循环引用的问题,即对象之间相互引用形成闭环,导致计数始终不为0。 2. 可达性分析算法(Reachability Analysis):JVM通过可达性分析算法,以GC Root对象作为起始点,沿着对象之间的引用链进行遍历,若某个对象GC Root对象不能形成引用链,则判断对象为不可达对象,即可以回收。 GC Root对象包括: - 虚拟机栈的引用:局部变量表的引用对象 - 方法区静态属性引用的对象:类的静态变量 - 方法区常量引用的对象:字符串常量等 - 本地方法栈的JNI引用:Java Native Interface引用的对象 3. 垃圾回收算法:JVM采用不同的垃圾回收算法来标记和回收可回收对象。常见的算法有标记清除算法(Mark and Sweep)、复制算法(Copying)、标记整理算法(Mark and Compact)等。这些算法会定期执行,对堆对象进行标记和回收,判断对象是否可以回收的依据是对象是否被标记为可达状态。 综上所述,JVM判断对象是否可以回收主要依靠垃圾回收算法和GC Root对象的可达性分析。当一个对象不再被引用或与GC Root对象没有引用链相连时,可以被判断可回收对象,并由垃圾回收器对其进行回收。 ### 回答3: JVM判断一个对象是否可以回收主要依赖两种垃圾回收算法:引用计数算法和可达性分析算法。 1. 引用计数算法:引用计数算法是通过为每个对象维护一个引用计数器来判断对象的引用数量。当对象被创建时,引用计数器初始化为1,当有新的引用指向对象时,引用计数器加1;当引用失效或超出作用域时,引用计数器减1。当引用计数器为0时,表示对象不再被引用,可以判定为可回收对象。 然而,引用计数算法不能解决循环引用的问题。当两个或多个对象互相引用时,它们的引用计数器都无法达到0,即使它们已经不再被程序使用,也不会被回收。 2. 可达性分析算法:可达性分析算法是JVM主要采用的垃圾回收算法,通过判断对象是否可被一组称为"GC Roots"的对象直接或间接引用来进行标记和回收。 在JVMGC Roots包括: - 当前执行方法的局部变量和输入参数所引用的对象 - 活动线程所引用的对象 - 静态变量所引用的对象 - JNI引用变量所引用的对象 GC Roots会被作为起始点,遍历对象引用关系图,将所有可达的对象标记为可达对象,而未被标记的对象即为不可达对象,也就是可回收对象。可达性分析算法能够解决循环引用的问题,因为循环引用的对象都不会被GC Roots直接或间接引用,将被判定为不可达对象。 当垃圾回收器运行时,会清除所有不可达的对象,释放其占用的内存空间。这样就判定了一个对象是否可以回收。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值