乐趣区

关于java:Java引用分类概述

Java变量的类型能够分为根本类型和援用类型。Java的援用类型能够划分为强脆弱虚四种。

强援用

程序中广泛呈现的,相似 “obj j = new Object()” 或者通过反射生成的对象。强援用指向的对象,只有可达性剖析中,对象在 援用链(Reference Chain)中,GC就不会对对象进行回收操作,哪怕会引起OOM

软援用(SoftReference)

形容一些还有用然而并非必须的对象。在零碎产生内存溢出之前,将会把这些对象列入回收范畴之中进行第二次回收(援用存在的状况下也会回收)。

public static void softTest() {
    // 生成一个占用 10m 内存的 软援用
    SoftReference softReference = new SoftReference(new byte[1024 * 1024 * 10]);
    System.out.println("before GC:" + softReference.get());
    System.gc();
    System.out.println("after  GC:" + softReference.get());
    // 在创立一个占用 10m 内存的 强援用对象
    byte[] obj = new byte[1024 * 1024 * 10];
    // 配置 -Xmx = 15m,在创立 obj 时堆内存不足(10 + 10 > 15)后,GC 回收 softReference 对象
    System.out.println("memory insufficient :" + softReference.get());
}

从执行后果可知,在堆内存短缺时,GC并不会被动回收软援用对象;而当堆内存不足时,GC会将软援用对象回收。

before GC: [B@7ba4f24f
after  GC: [B@7ba4f24f
memory insufficient : null

用处 :依据软援用的特点,能够用于 缓存 一些大的对象、图片。

弱援用(WeakReference)

也是形容非必须对象的,然而比软援用强度更弱,被弱援用的对象只能生存到下一次垃圾回收之前。只有弱援用的 对象在 GC 产生后间接被回收。

public static void weakTest() {WeakReference weakReference = new WeakReference(new byte[1024 * 1024 * 10]);
    System.out.println("before GC:" + weakReference.get());
    System.gc();
    System.out.println("after  GC:" + weakReference.get());
}

从执行后果可知,弱援用会在 GC 时被间接回收。

before GC: [B@7ba4f24f
after  GC: null

用处 :只存在弱援用的对象会在GC 时被间接回收,能够通过搭配弱援用和强援用,实现一些对象在不应用后主动进行回收。WeakHashMap、ThreadLocal都应用了WeakReference

虚援用(PhantomReference)

也成幽灵援用或者幻影援用,一个对象是否有虚援用的存在,齐全不对其造成影响。虚援用无论是否进行过 GC 都无奈获取其值。

public static void phantomTest() throws InterruptedException {ReferenceQueue queue = new ReferenceQueue();
    PhantomReference phantomReference = new PhantomReference(new byte[1024 * 1024 * 10], queue);
    System.out.println("before GC:" + phantomReference.get());
    System.gc();
    System.out.println("after  GC:" + phantomReference.get());
} 
before GC: null
after  GC: null

用处:虚援用个别用于 NIO 操作间接内存。

Reference

脆弱虚援用,都继承了 Reference 抽象类。Reference规定了援用的根本规定。

  • Reference内的对象寄存在 referent 变量中
  • 能够绑定一个ReferenceQueue
  • 对象被回收是指 referent 变量指向的对象,而不是 Reference 对象

无论是软援用、弱援用还是虚援用,都能够绑定一个 ReferenceQueue 对象,当援用被回收后,会将此 援用 putReferenceQueue中,开发人员能够通过监听 ReferenceQueue 进行一些扩大操作。WeakHashMap 就是通过此形式解决曾经被删除 keyvalue的。

public static void weakTest() throws InterruptedException {ReferenceQueue queue = new ReferenceQueue();
    WeakReference weakReference = new WeakReference(new byte[1024 * 1024 * 10], queue);
    System.out.println("before GC:" + weakReference.get() + "| queue value:" + queue.remove(100));
    System.gc();
    System.out.println("after  GC:" + weakReference.get() + "| queue value:" + queue.remove(100));
}

通过运行后果能够看出,在 WeakReference 被回收后,就被 put 进入了 queue 中。

before GC: [B@7ba4f24f | queue value: null
after  GC: null | queue value: java.lang.ref.WeakReference@3b9a45b3

参考文档

https://mp.weixin.qq.com/s/HF…

退出移动版