JVM之判断对象是否可以被回收

1.引用计数算法

在对象中添加一个引用计数器,如果该对象被引用一次,计数器就加一,如果引用失效一个,计数器就减一;如果计数器为零,则该对象就可以被回收。

​ 在Java中一般不采用引用计数算法对对象进行管理,因为引用计数算法需要考虑一些特殊情况,必须要配合大量额外工作才能保证正确地工作,例如典型的对象之间互相循环引用的问题。如果两个对象之间互相引用,会导致这两个对象的计数器始终不为零,最终导致两个对象永远无法被清理的情况(类似于死锁)。

2.可达性分析算法

​ 主流的商用程序语言(Java、C#等)都采用可达性分析算法来判断对象是否要被回收。

该算法通过一系列的 “GC Roots” 的根对象作为起始的节点集,然后从这些节点开始,根据引用关系向下索引,通过对象之间的引用关系(“引用链”)遍历所有跟 “GC Roots” 相关的对象。而与 “GC Roots” 没有用任何引用链相连的对象,用图论的说法就是从GC Roots到该对象不可达,即此对象为可以回收的对象。

​ 可作为GC Roots的对象一般包括:

  • 虚拟机栈中局部变量表中引用的对象
  • 本地方法栈中JNI(即本地方法)所引用的对象
  • 方法区中静态属性引用的对象(如静态变量的引用)
  • 方法区中常量引用的对象(如字符串常量池中的引用)
3.引用类型

​ 在JDK1.2版以前,Java对引用的定义很简单,只有 “被引用” 和 “未被引用” 两种状态。在JDK1.2之后,Java对引用的概念进行了扩充,将引用分为了强引用软引用弱引用虚引用

  • 强引用:类似于 new 一个新对象的引用。使用强引用相关联的对象永远不会被回收。

  • 软引用:用来描述一些还有用但非必须的对象。使用软引用相关联的对象,只有在内存不够的时候,才会进行清理。

  • 弱引用:描述非必须对象,只能存活到下一次垃圾回收。

  • 虚引用:虚引用不会对对象的生存时间造成影响,也无法通过虚引用获得一个对象实例。设置虚引用的唯一目的就是为了在这个对象被回收的时候收到一个系统通知。

4.finalize( )方法

​ 即使对象被可达性分析算法判定为可以回收的对象,该对象仍然有自救的方法。如果该对象覆盖了finalize()方法,且该方法没有被虚拟机调用过,在该对象被判定为可回收对象时,会被放置在一个回收队列中,如果对象在该队列中,被某个引用链上的对象重新关联,便可以逃脱被回收的命运。但是这种自救只能进行一次,如果该对象再次被作为清理的对象,将不再调用finalize()方法。

​ 在Java中已经不再推荐使用该方法,因为回收队列还是位于堆中,如果大量的对象进入回收队列中,势必会对堆内存有很大的占用,严重的会导致内存溢出。


注:本文参考自《深入理解Java虚拟机》第三版

  • 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"的对象直接或间接引用来进行标记和回收。 在JVM,GC Roots包括: - 当前执行方法的局部变量和输入参数所引用的对象 - 活动线程所引用的对象 - 静态变量所引用的对象 - JNI引用变量所引用的对象 GC Roots会被作为起始点,遍历对象引用关系图,将所有可达的对象标记为可达对象,而未被标记的对象即为不可达对象,也就是可回收对象。可达性分析算法能够解决循环引用的问题,因为循环引用的对象都不会被GC Roots直接或间接引用,将被判定为不可达对象。 当垃圾回收器运行时,会清除所有不可达的对象,释放其占用的内存空间。这样就判定了一个对象是否可以回收
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值