在JDK1.2之后,Java对引用的概念进行了扩充,将引用分为强引用(Strongly Reference)、软引用(Soft Reference)、弱引用(Weak Reference)和虚引用(Phantom Reference)4种,这4种引用强度依次逐渐减弱。
一、强引用(Strongly Reference)
强引用是最传统的“引用”的定义,是指在程序代码之中普遍存在的引用赋值,即类似“Object obj = new Object()”这种引用关系。无论任何情况下,只要强引用关系还存在,垃圾收集器就永远不会回收掉被引用的对象。
二、软引用(Soft Reference)
软引用是用来描述一些还有用,但非必须的对象。只被软引用关联着的对象,在系统将要发生内存溢出异常前,会把这些对象列进回收范围之中进行第二次回收,如果这次回收还没有足够的内存,才会抛出内存溢出异常。在JDK1.2版之后提供了SoftReference类来实现软引用。
三、弱引用(Weak Reference)
弱引用也是用来描述那些非必须对象,但是它的强度比软引用更弱一些,被引用关联的对象只能生存到下一次垃圾收集发生为止。当垃圾收集器开始工作,无论当前内存是否足够,都会回收掉只被弱引用关联的对象。在JDK1.2版之后提供了WeakReference类来实现弱引用。
四、虚引用(Phantom Reference)
虚引用也成为“幽灵引用”或者“幻影引用”,她是最弱的一种引用关系。一个对象是否有虚引用的存在,完全不会对其生存时间构成影响,也无法通过虚引用来取得一个对象实例。为一个对象设置虚引用关联的唯一目的只是为了能在这个对象被收集器回收时收到一个系统通知。在JDK1.2之后提供了PhantomReference类来实现虚引用。
注意:
(1)DirectByteBuffer(可参考:https://www.sohu.com/a/167784796_355142)是通过虚引用(Phantom Reference)来实现堆外内存的释放的。
(2)虚引用(Phantom Reference)不同于软引用和弱引用,虚引用无法通过 get() 方法来取得目标对象的强引用从而使用目标对象,观察源码可以发现 get() 被重写为永远返回 null。
(3)虚引用到底有什么作用?
其实虚引用主要被用来跟踪对象被垃圾回收的状态,通过查看引用队列(垃圾回收时会将要回收的对象放到一个引用队列ReferenceQueue中)中是否包含对象所对应的虚引用来判断它是否即将被垃圾回收,从而采取行动。它并不被期待用来取得目标对象的引用,而目标对象被回收前,它的引用会被放入一个ReferenceQueue对象中,从而达到跟踪对象垃圾回收的作用。
堆外内存就是用到了虚引用,可参考《Java中的堆外内存和堆内内存》。
梧高凤必至,花香蝶自来。