JVM判定对象是否存活

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接: https://blog.csdn.net/u013632755/article/details/99372776

判定对象是否存活算法

1、引用计数算法

给对象中添加一个引用计数器,每当一个地方引用它时,计数器值就+1;当引用失效时,计数器值就-1;任何时刻计数器为0的对象就是不可能再被使用的。
优点:实现简单,判定效率高
缺点:很难解决对象相互循环引用问题

对于JVM,并不是使用此算法,使用的是第二种算法:可达性分析算法。
如下示例看GC日志发现,objA和objB相互引用,显示置为null之后,进行GC时不会因为相互引用而不回收objA和objB,所以证明JVM并不是使用的引用计数算法。

public class ReferenceCountingGc{
    public Object instance=null;
    private static final int _1MB=1024*1024;
    
    private byte[] bigSize= new byte[2*_1MB];
    
    public static void testGC(){
        ReferenceCountingGC objA=new ReferenceCountingGC();
        ReferenceCountingGC objB=new ReferenceCountingGC();
        objA.instance=objB;
        objB.instance=objA;
        
        objA=null;
        objB=null;
        
        System.gc();
    }
}
2、可达性分析算法

在主流商用程序语言(Java、C#)的主流实现中,都是通过可达性分析来判定对象是否存活的。
基本思路是通过一系列称为"GC Roots"的对象作为起始点,从节点开始向下搜索,搜索所走过的路径成为引用链,当一个对象到GC Roots没有任何引用链相连时,则证明此对象时不可用的。

在java语言中,可作为GC Roots的对象包括下面几种:

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

JDK引用

强引用
强引用指类似 Object obj=new Object()这类的引用,只要强引用还在,垃圾收集器永远不会回收掉被引用的对象。
软引用
软引用是用来描述一些还有用但并非必需的对象。在系统将要发生内存溢出异常之前,将会把软引用对象列进回收范围进行二次回收。如果这次回收之后还是没有足够内存,才会抛出内存溢出异常。在JDK1.2之后,提供了SoftReference类来实现软引用。
弱引用
弱引用关联的对象只能存活到下一次GC发生之前。当GC开始时,无论内存是否足够,都会回收弱引用关联的对象。在JDK1.2之后,提供了WeakReference类实现。
虚引用
一个对象是否有虚引用的存在,完全不会对其生存时间构成影响,也无法通过虚引用来取得一个对象实例。虚引用的唯一目的是在这个对象被GC时收到一个系统通知。在JDK1.2之后,提供了PhantomReference类实现。

猜你喜欢

转载自blog.csdn.net/u013632755/article/details/99372776