Java的强、软、弱、虚引用介绍与分析

Java引用类型

Java引用主要分为4种(其实似乎是5种):

  • Strong Reference 强引用,直接引用
  • Soft Reference 软引用,间接引用
  • Weak Reference 弱引用,间接引用
  • Phantom Reference 虚引用,几乎无引用
  • Final引用,这里不介绍

强引用

Object strongReference = new Object();

我们平常使用最多的就是强引用。按照JVM规范,在GC时通过可达性分析检测到强引用可达时,这个对象不会被回收。
但是在某些情况下,强引用的这个特性会引起OutOfMemoryError,比如一直向集合中添加元素。

软引用
软引用对象仅在内存不足时会被回收,JVM保证在抛出OutOfMemoryError之前已经将软引用对象全部清理了。

弱引用
JVM不保证弱引用对象的回收时机,在GC线程发现该对象为弱引用对象时就会回收。

虚引用
任何时候都可能会被回收。

Java提供了几种间接引用方式,有什么好处吗?是如何起作用的?我们从源码中一点一点看。

Reference和ReferenceQueue

ReferenceQueue
ReferenceQueue本质上是一个单向链表,链表的节点是Reference,并提供了同步非阻塞出队方法poll()
同时,ReferenceQueue提供了一个类似BlockingQueue的阻塞获取节点的方法remove()
在Reference类的构造方法中,可以指定一个ReferenceQueue

Reference
Java的强软弱虚四种中,软、弱、虚引用都是继承Reference<T>抽象类。
核心的几个属性和方法:

public abstract class Reference<T> {
    ......
    private T referent; // 代管的对象
    ......
    volatile ReferenceQueue<? super T> queue;   // 关联的通知队列
    ......
    private static Reference<Object> pending = null;    // 静态的待处理链
    ......
    static boolean tryHandlePending(boolean waitForNotify) {......} // 负责检查pending链等关键操作
    ......
    public T get() { return this.referent; }    // 获取直接引用
    ......
}

Reference类相当于引用代理,正常情况下我们持有Reference对象的引用,Reference对象持有我们真正使用的对象T referent的引用。
Reference提供了几个重要的方法:

  • get() 获取代管对象(referent)的直接引用
  • isEnqueued() 检查是否还在队列中
  • tryHandlePending() 如果存在可回收的Reference,则将这个Reference插入到构造方法指定的ReferenceQueue中;否则wait()等待

其中,Reference创建了一个名为“Reference Handler”的常驻后台线程,用于将referent已被回收的Reference对象加入其引用通知队列ReferenceQueue。

Reference中的referent
Reference中的referent会受到GC的特殊对待:

  • GC检测到reference.referent(非静态)引用可达性发生变化时,会将对应的Reference对象挂在Reference.pending(静态)链上

ps.文档未介绍“可达性发生变化”是什么情况,我猜测是处于“仅Reference对象持有直接引用”的状态。这个状态按理说是不能回收的,因为引用分析是可达的,但这里特殊处理,即使可达GC也可以回收。

综合看Reference和ReferenceQueue
这两个类形成了一个回收通知机制:

  1. 新建一个Reference对象,在构造方法中为其绑定一个间接引用对象referent,及一个回收通知队列referenceQueue
  2. 去除referent的其他直接引用
  3. Reference类在应用启动时就创建了Reference Handler线程,但Reference.tryHandlePending方法发现pending对象为null,线程进入WAIT状态,pending保存的是需要回收的Reference链的首个节点
  4. GC在回收这个Reference对象的referent时,将其Reference对象挂在Reference.pending链上,即GC会将所有达到可回收状态的Reference对象挂到静态的pending链上
  5. Reference Handler线程被唤醒,通过Reference.tryHandlePending方法发现pending中有内容,将链上的Reference对象插入其构造方法指定的referenceQueue中(当然,如果没有指定那么就不会入队)
  6. 我们通过referenceQueue的remove()阻塞poll()非阻塞方法可以拿到哪些Reference对象的referent已被回收,可以进行接下来的善后处理
参考资料中的图

应用场景

  • 软引用
    官方文档提到,软引用非常适合用于内存敏感的缓存,当内存不足时GC将会回收缓存中的部分内容。
    ps.有参考资料提到软引用可能导致频繁Full GC。
  • 弱引用
    官方文档提到,弱引用适合用于规范化映射(Canonicalizing Mappings),可以理解为一个Map中的KV映射。
  • 虚引用
    PhantomReference.get()方法永远返回null,因此一旦失去直接引用仅保留虚引用,那么就无法再获取这个对象的引用。适合用于监控对象是否被回收。

一个典型的应用就是WeakHashMap类。
WeakHashMap中的Entry继承自WeakReference类,其中key就是间接引用中的referent,获取key均是通过Reference对象的get()方法,即有可能失效。
另外,WeakHashMap持有一个ReferenceQueue对象,每个Entry都在构造方法中关联了这个queue。
expungeStaleEntries()方法会清理失效的Entry,作为大多数方法的前置操作。

结合上边的回收通知机制等分析,WeakHashMap有这些特点:

  1. 使用弱引用作为Entry,referent是Entry的key,并关联了一个内部的ReferenceQueue
  2. 每次GC时都会清理掉一部分Entry的referent,key变为null——相当于这些内容在Map中“消失”了,不过value占用的空间还没清理
  3. key被清理的Entry,如前所述会进入回收通知队列ReferenceQueue
  4. 在绝大部分操作之前会先从回收通知队列中获取失效的Entry,并从Map中真正删除

回收通知机制的好处

  1. GC虽然做了特殊处理,但是插入链表这个额外的操作相对来说负担是非常小的,对GC影响不大,回收通知机制后续的逻辑其实是Reference Handler线程在做。
  2. 使用间接引用后,如何清理被回收的数据及相关数据,或者如何接收被回收通知是个问题。
    以WeakHashMap举例,被回收的数据都在ReferenceQueue中,比起遍历整个集合查看哪些数据被回收了,显然使用通知机制仅查这个queue效率要高得多。

参考资料

JDK源码阅读-Reference
Java Reference详解 - robin-yao的个人页面 - OSCHINA
Java 理论与实践: 用弱引用堵住内存泄漏

本文搬自我的博客,欢迎参观!

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

推荐阅读更多精彩内容