Java四种引用类型

介绍

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

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

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

image.png

StrongReference

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

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

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

image.png
StrongReference的特性

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


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

引用队列(ReferenceQueue)

在很多场景下,我们的程序需要在一个对象的可达性(是否已经被GC回收)发生变化时得到通知,引用队列就是用于收集这些信息的队列。在创建SoftReference对象时,可以为其关联一个引用队列,当SoftReference所引用的对象被GC回收时,Java虚拟机就会将该SoftReference对象添加到与之关联的引用队列中。当需要检测这些通知信息时,就可以从引用队列中获取这些SoftReference对象。不仅是SoftReference,下面介绍的弱引用和虚引用都可以关联相应的队列。

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是典型的弱引用例子。

Entry弱引用key,强引用value;当不再由强引用指向key时,则key可以被垃圾回收,当key被垃圾回收之后,对应的Entry对象会被Java虚拟机加入到其关联的队列中。当应用程序下次操作WeakHashMap时,例如对WeakHashMap的扩容操作,就会遍历关联的引用队列,将其中的Entry对象从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在系统中是StrongReference强引用的, 那么WeakHashMap将退化为一个普通的HashMap,因为它不能被GC回收。

image.png

参考资料

再谈四种引用状态

常用JVM命令参数

Do You Really Know the 4 Reference Types in Java?

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,456评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,370评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,337评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,583评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,596评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,572评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,936评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,595评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,850评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,601评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,685评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,371评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,951评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,934评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,167评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,636评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,411评论 2 342

推荐阅读更多精彩内容