Java 四种引用类型

本贴最后更新于 1699 天前,其中的信息可能已经水流花落

介绍

Java 中提供了四种引用类型,分别如下:

  • FinalReference(强引用)
  • SoftReference(软引用)
  • WeakReference(弱引用)
  • PhantomReference(虚引用)

其中 FinalReference 是包权限无法使用,其它三种引用类型都是公共的可以在应用中使用,下面是 Reference 的类结构。

091F88D366BB4A199B5F8F4CC4C9CD95.png

FinalReference

Java 中的强引用其实就是 new 对象,可以通过引用操作堆中的对象(和 C 中的指针类似),例如:

StringBuffer buffer = new StringBuffer("HelloWorld!");

变量 buffer 指向 StringBuffer 所在的堆空间,通过 buffer 来进行操作。

03CE86AECCB14A8489DE6904D3760209.png

FinalReference 的特性

1.可以直接访问目标对象;
2.指向的对象不会被 GC 回收 ♻️,当 JVM 内存不足时会抛出 OOM 异常终端程序;
3.基于上面第 2 点,当其它需要释放的代码块持 FinalReference 会造成内存泄露;


SoftReference

SoftReference 即软引用,当 JVM 堆空间使用率到达阈值的时候会触发 GC 回收,我们可以用它来实现对内存敏感的缓存,SoftReference 的特性是可以保留对 Java 对象软引用的实例,软引用的实例并不会阻止 GC 进行回收,在 GC 线程进行回收之前我们可以通过它的 get 方法获取到对象的强引用,一旦对象被 GC 回收后 get 方法将返回 null。

下面我们通过代码来实践,通过设置 JVM 堆内存大小为 2M 来模拟:

java -Xms2M -Xmx2M -verbose:gc -XX:+PrintGCDetails [class文件]
  • -Xms2M
    堆大小固定为 2M
  • -verbose:gc
    输出虚拟机中 GC 的详细情况
  • -XX:+PrintGCDetails
    在控制台上打印出 GC 具体细节
    public static void testSoftRef() {
        SoftReference<byte[]> softRef1 = new SoftReference<>(new byte[1024 * 300]);
        SoftReference<byte[]> softRef2 = new SoftReference<>(new byte[1024 * 300]);
        SoftReference<byte[]> softRef3 = new SoftReference<>(new byte[1024 * 300]);
        SoftReference<byte[]> softRef4 = new SoftReference<>(new byte[1024 * 300]);
        SoftReference<byte[]> softRef5 = new SoftReference<>(new byte[1024 * 300]);

        System.out.println(softRef1.get());
        System.out.println(softRef2.get());
        System.out.println(softRef3.get());
        System.out.println(softRef4.get());
        System.out.println(softRef5.get());
    }

结果如下,我们可以看到上面分配的 byte 长度是超出年轻代大小的,当内存不足时的确触发了 GC 进行回收,正如上面所说的那样。

[GC (Allocation Failure) [PSYoungGen: 510K->320K(1024K)] 510K->320K(1536K), 0.0037082 secs] [Times: user=0.01 sys=0.00, real=0.00 secs] 
[GC (Allocation Failure) [PSYoungGen: 825K->432K(1024K)] 825K->432K(1536K), 0.0016384 secs] [Times: user=0.00 sys=0.00, real=0.00 secs] 
[GC (Allocation Failure) [PSYoungGen: 942K->496K(1024K)] 942K->512K(1536K), 0.0113468 secs] [Times: user=0.01 sys=0.00, real=0.01 secs] 
[GC (Allocation Failure) --[PSYoungGen: 984K->984K(1024K)] 1300K->1493K(1536K), 0.0014734 secs] [Times: user=0.00 sys=0.01, real=0.00 secs] 
[Full GC (Ergonomics) [PSYoungGen: 984K->34K(1024K)] [ParOldGen: 509K->493K(512K)] 1493K->527K(1536K), [Metaspace: 3282K->3282K(1056768K)], 0.0131428 secs] [Times: user=0.01 sys=0.00, real=0.02 secs] 
[Full GC (Ergonomics) [PSYoungGen: 334K->334K(1024K)] [ParOldGen: 493K->492K(512K)] 827K->827K(1536K), [Metaspace: 3282K->3282K(1056768K)], 0.0106514 secs] [Times: user=0.00 sys=0.00, real=0.01 secs] 
[Full GC (Allocation Failure) [PSYoungGen: 334K->0K(1024K)] [ParOldGen: 492K->397K(512K)] 827K->397K(1536K), [Metaspace: 3282K->3282K(1056768K)], 0.0042212 secs] [Times: user=0.01 sys=0.00, real=0.00 secs] 
[GC (Allocation Failure) [PSYoungGen: 300K->332K(1024K)] 697K->729K(1536K), 0.0002648 secs] [Times: user=0.00 sys=0.00, real=0.00 secs] 
null
null
null
[B@610455d6
[B@511d50c0
Heap
 PSYoungGen      total 1024K, used 646K [0x00000007bfe80000, 0x00000007c0000000, 0x00000007c0000000)
  eden space 512K, 61% used [0x00000007bfe80000,0x00000007bfecea50,0x00000007bff00000)
  from space 512K, 64% used [0x00000007bff80000,0x00000007bffd3010,0x00000007c0000000)
  to   space 512K, 0% used [0x00000007bff00000,0x00000007bff00000,0x00000007bff80000)
 ParOldGen       total 512K, used 397K [0x00000007bfe00000, 0x00000007bfe80000, 0x00000007bfe80000)
  object space 512K, 77% used [0x00000007bfe00000,0x00000007bfe63570,0x00000007bfe80000)
 Metaspace       used 3291K, capacity 4500K, committed 4864K, reserved 1056768K
  class space    used 363K, capacity 388K, committed 512K, reserved 1048576K

WeakReference

WeakReference 即弱引用,当触发 GC 时,无论 JVM 堆内存是否足够对象都会被回收,下面进行测试:

    public static void testWeakRef() {
        byte[] buffer = new byte[1024 * 500];
        WeakReference<byte[]> weakReference = new WeakReference<>(buffer);
        System.out.println("GC前:" + weakReference.get());
        buffer = null;
        //手动触发GC
        System.gc();
        System.out.println("GC后:" + weakReference.get());
    }

结果如下:

GC前:[B@610455d6
GC后:null

SoftReference 和 WeakReference 都适用于保存可选的缓存数据,在系统内存不足时,将回收缓存的数据不会导致 OOM,并且缓存也能存在很长一段时间。

PhantomReference

PhantomReference 即虚引用,是所有类型中最弱的,它几乎是没有引用因为随时会被 GC 回收,当调用它的 get 方法获取强引用时始终都是返回 null,它必须要和 ReferenceQueue 一起使用,用来跟踪垃圾回收过程。

当 GC 要回收对象时,如果发现 PhantomReference 后将进行 GC 然后销毁该对象,并将 PhantomReference 添加到 ReferenceQueue 中,判断是否向 ReferenceQueue 添加了 PhantomReference 可以确定是否需要对引用的对象进行回收,如果 ReferenceQueue 中存在 PhantomReference 那么在回收引用对象之前可以进行一些额外的操作。

    public static void testPhantomRef() {
        byte[] buffer = new byte[1024 * 500];
        ReferenceQueue<Object> referenceQueue = new ReferenceQueue<>();
        PhantomReference<byte[]> phantomReference = new PhantomReference<>(buffer,referenceQueue);
        buffer = null;
        System.out.println("GC前:" + phantomReference.get());
        System.gc();
        System.out.println("GC后:" + phantomReference.get());
    }

结果如下:

GC前:null
GC后:null

关于 PhantomReference 的 get 方法总是返回 null。

    /**
     * Returns this reference object's referent.  Because the referent of a
     * phantom reference is always inaccessible, this method always returns
     * <code>null</code>.
     *
     * @return  <code>null</code>
     */
    public T get() {
        return null;
    }

WeakHashMap

顾名思义,它和 HashMap 一样都是实现了 Map 接口,只不过它使用的是 WeakReference 作为存储,WeakHashMap 是典型的弱引用例子。

public class WeakHashMap<K,V>
    extends AbstractMap<K,V>
    implements Map<K,V> {}

private static class Entry<K,V> extends WeakReference<Object> implements Map.Entry<K,V> {}

需要注意的是如果 WeakHashMap 的 key 在系统中是 FinalReference 强引用的, 那么 WeakHashMap 将退化为一个普通的 HashMap,因为它不能被 GC 回收。

参考资料

再谈四种引用状态

常用 JVM 命令参数

Do You Really Know the 4 Reference Types in Java?

  • Java

    Java 是一种可以撰写跨平台应用软件的面向对象的程序设计语言,是由 Sun Microsystems 公司于 1995 年 5 月推出的。Java 技术具有卓越的通用性、高效性、平台移植性和安全性。

    3190 引用 • 8214 回帖 • 1 关注
1 操作
xumiao 在 2020-04-28 21:10:45 更新了该帖

相关帖子

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...