首页 > 代码库 > java中虚引用PhantomReference与弱引用WeakReference(软引用SoftReference)的差别

java中虚引用PhantomReference与弱引用WeakReference(软引用SoftReference)的差别

之前的这篇博客介绍了java中4种引用的差别和使用场景,在最后的总结中提到:

软引用和弱引用差别不大,JVM都是先把SoftReference和WeakReference中的referent字段值设置成null,之后加入到引用队列;而虚引用则不同,如果某个堆中的对象,只有虚引用,那么JVM会将PhantomReference加入到引用队列中,JVM不会自动将referent字段值设置成null”。这段总结写的比较仓促,也没有给出实际的例子加以佐证。本文主要是重申下这几种引用的差别,并给出实际的例子,让读者清楚的感受到它们的差别。


软引用和弱引用差别不大,JVM都是先将其referent字段设置成null,之后将软引用或弱引用,加入到关联的引用队列中。我们可以认为JVM先回收堆对象占用的内存,然后才将软引用或弱引用加入到引用队列

而虚引用则不同,JVM不会自动将虚引用的referent字段设置成null,而是先保留堆对象的内存空间,直接将PhantomReference加入到关联的引用队列,也就是说如果我们不手动调用PhantomReference.clear(),虚引用指向的堆对象内存是不会被释放的。


referent是java.lang.ref.Reference类的私有字段,虽然没有暴露出共有API来访问这个字段,但是我们可以通过反射拿到这个字段的值,这样就能知道引用被加入到引用队列的时候,referent到底是不是null。SoftReference和WeakReference是一样的,这里我们以WeakReference为例。

package ref.referent;

import java.lang.ref.Reference;
import java.lang.ref.ReferenceQueue;
import java.lang.ref.WeakReference;
import java.lang.reflect.Field;

// 会报空指针:WeakReference中的referent被设置成null,之后加入到ReferenceQueue
public class TestWeakReference
{
	private static volatile boolean isRun = true;

	private static volatile ReferenceQueue<String> referenceQueue = new ReferenceQueue<String>();

	public static void main(String[] args) throws Exception
	{
		String abc = new String("abc");
		System.out.println(abc.getClass() + "@" + abc.hashCode());

		new Thread() {
			public void run()
			{
				while (isRun)
				{
					Object o = referenceQueue.poll();
					if (o != null)
					{
						try
						{
							Field rereferent = Reference.class
									.getDeclaredField("referent");
							rereferent.setAccessible(true);
							Object result = rereferent.get(o);
							System.out.println("gc will collect:"
									+ result.getClass() + "@"
									+ result.hashCode());
						} catch (Exception e)
						{
							e.printStackTrace();
						}
					}
				}
			}
		}.start();

		// 对象是弱可达的
		WeakReference<String> weak = new WeakReference<String>(abc,
				referenceQueue);
		System.out.println("weak=" + weak);

		// 清除强引用,触发GC
		abc = null;
		System.gc();

		Thread.sleep(3000);
		isRun = false;
	}
}
运行这段代码会发现,我们创建的Thread中报空指针异常。当我们清除强引用,触发GC的时候,JVM检测到new String("abc")这个堆中的对象只有WeakReference,那么JVM会释放堆对象的内存,并自动将WeakReference的referent字段设置成null,所以result.getClass()会报空指针异常。


代码与上面类似, 我们将WeakReference替换成PhantomReference:

package ref.referent;

import java.lang.ref.PhantomReference;
import java.lang.ref.Reference;
import java.lang.ref.ReferenceQueue;
import java.lang.reflect.Field;

// 当PhantomReference加入到ReferenceQueue的时候,目标对象内存空间仍然存在不会被回收.
// PhantomReference中的referent字段不会被JVM自动设置成null
// 当目标对象的PhantomReference加入到ReferenceQueue的时,此时目标对象是强可达的
public class TestPhantomReference
{
	private static volatile boolean isRun = true;

	private static volatile ReferenceQueue<String> referenceQueue = new ReferenceQueue<String>();

	public static void main(String[] args) throws Exception
	{
		String abc = new String("abc");
		System.out.println(abc.getClass() + "@" + abc.hashCode());

		new Thread() {
			public void run()
			{
				while (isRun)
				{
					Object o = referenceQueue.poll();
					if (o != null)
					{
						try
						{
							Field rereferent = Reference.class
									.getDeclaredField("referent");
							rereferent.setAccessible(true);
							Object result = rereferent.get(o);
							System.out.println("gc will collect:"
									+ result.getClass() + "@"
									+ result.hashCode());
						} catch (Exception e)
						{
							e.printStackTrace();
						}
					}
				}
			}
		}.start();

		// 测试情况1:对象是虚可达的
		PhantomReference<String> phantom = new PhantomReference<String>(abc,
				referenceQueue);
		System.out.println("phantom=" + phantom);

		// 测试情况2:对象是不可达的,直接就被回收了,不会加入到引用队列
		// new PhantomReference<String>(abc, referenceQueue);

		// 清除强引用,触发GC
		abc = null;
		System.gc();

		Thread.sleep(3000);
		isRun = false;
	}
}
运行这段代码会发现,程序没有报异常,执行结果是:

class java.lang.String@96354
phantom=java.lang.ref.PhantomReference@15b7986
gc will collect:class java.lang.String@96354

很明显,当PhantomReference加入到引用队列的时候,referent字段的值并不是null,而且堆对象占用的内存空间仍然存在。也就是说对于虚引用,JVM是先将其加入引用队列,当我们从引用队列删除PhantomReference对象之后(此时堆中的对象是unreachable的),那么JVM才会释放堆对象占用的内存空间。由此可见,使用虚引用有潜在的内存泄露风险,因为JVM不会自动帮助我们释放,我们必须要保证它指向的堆对象是不可达的。从这点来看,虚引用其实就是强引用,当内存不足的时候,JVM不会自动释放堆对象占用的内存。后续的帖子我会进行一些OOM相关的实验,去证明虚引用的确会导致OOM,而软引用和弱引用则不会导致OOM。


小结:

上面的测试代码,只是为了帮助我们看清楚虚引用与软引用/弱引用的不同表现。在实际的开发中,我们是不会通过反射获取referent字段的值,这样做毫无意义,也不值得提倡。


java中虚引用PhantomReference与弱引用WeakReference(软引用SoftReference)的差别