前言
很早之前就有看过一些关于Java 引用的文章,此次借着阅读<<深入理解Java虚拟机>>一书,归纳引用这一块的一些知识。而关于为何要区分这么些个引用呢?我的理解是帮助GC(垃圾回收器)回收对象,达到管理内存的目的。
强引用
类似 Object obj=new Object()
的引用方式,GC(垃圾收集器)永远不会回收。可以通过显示的置为null,来帮助垃圾收集器回收此对象。这也是我们在阅读一些JDK的集合源码时,有时会看到大神们似乎"多此一举"的将一个对象显示置为null,并且后边还有这样的注释:// help GC
软引用
描述一些还有用,但非必须的对象。在系统将要发生内存溢出异常前,会把被软引用关联着的对象列进回收范围之中,进行第二次回收,如果这次回收还没有足够的内存,才会抛出内存溢出异常。在JDK 1.2版之后提供了SoftReference
类来实现软引用。
场景:
- 一些不痛不痒的缓存,如网页缓存、图片缓存等
Persion persion = new Persion();
// 置为软引用
SoftReference<Persion> softReference = new SoftReference<>(persion);
if (softReference.get() != null) {
// 垃圾回收器还没有回收, 直接返回
softReference.get();
}
弱引用
同样用来描述那些非必须对象,但是它的强度比软引用更弱,被弱引用关联的对象只能生存到下一次垃圾收集发生为止。当垃圾收集器开始工作,无论当前内存是否足够,都会回收掉只被弱引用关联的对象。在JDK 1.2版之后提供了WeakReference
类来实现弱引用。
场景:
- 没想出来。感觉除了在一些组件源码中有看到使用(如ThreadLocal),现实生活场景还没能运用上。
- 不过有个万金油的说法,如果这个对象是偶尔的使用,并且希望在使用时随时就能获取到,但又不想影响此对象的垃圾收集,那么你应该用 Weak Reference 来记住此对象。
虚引用
虚引用又称为“幽灵引用”或者“幻影引用”,顾名思义,形同虚设,最弱的一种引用关系。一个对象是否有虚引用的存在,完全不会对其生存时间构成影响,也无法通过虚引用来取得一个对象实例。
为一个对象设置虚引用关联的唯一目的只是为了能在这个对象被收集器回收时收到一个系统通知。(譬如知道具体何时其引用的对象从内存中移除)在JDK 1.2版之后提供了PhantomReference
类来实现虚引用。