0

0

什么是Java中的引用队列(ReferenceQueue)_配合虚引用进行资源回收

P粉602998670

P粉602998670

发布时间:2026-02-11 14:21:09

|

531人浏览过

|

来源于php中文网

原创

虚引用必须搭配ReferenceQueue才能感知对象被回收,因其本身不阻止回收且get()恒返回null;JVM通过Reference Handler线程将已回收对象对应的虚引用入队,需主动poll()或remove()监听,否则无法获知回收事件。

什么是java中的引用队列(referencequeue)_配合虚引用进行资源回收

虚引用为什么必须搭配 ReferenceQueue 才能感知对象被回收

虚引用(PhantomReference)本身不能阻止对象被回收,也不提供任何访问对象的途径——它唯一的作用就是“通知你:这东西刚被 GC 清掉了”。但 JVM 不会主动调用你的代码,所以得靠 ReferenceQueue 来被动监听。

常见错误现象:PhantomReference.get() 永远返回 null;不把虚引用注册到队列里,就永远等不到回收通知。

  • 创建虚引用时,第二个参数必须是已初始化的 ReferenceQueue 实例,否则等于白建
  • 回收通知不是实时的:GC 后对象进入“pending”状态,由 JVM 的 Reference Handler 线程异步入队,可能有毫秒级延迟
  • 必须轮询 queue.poll() 或阻塞调用 queue.remove() 才能拿到被回收的虚引用对象(注意:不是原对象!是那个 PhantomReference 实例)

ReferenceQueue 的实际使用姿势:别直接 new,要配合清理逻辑

很多人以为把虚引用丢进队列就完事了,其实关键在“谁来消费队列”。JVM 只负责把虚引用对象放进去,后续资源释放、日志记录、连接关闭等,全得你自己写逻辑去处理。

使用场景:文件句柄、Socket 连接、JNI 分配的内存、自定义缓存对象等无法靠 finalize() 安全清理的资源。

立即学习Java免费学习笔记(深入)”;

  • 不要在 ReferenceQueue 上做耗时操作(比如网络请求),否则会卡住 Reference Handler 线程,影响其他引用处理
  • 推荐用单独线程 + queue.remove() 阻塞等待,避免空轮询浪费 CPU
  • 每次从队列取出 PhantomReference 后,应立即执行清理,并考虑是否需要显式调用 clear()(虽然虚引用本身已不可达,但保持习惯更安全)
ReferenceQueue queue = new ReferenceQueue<>();
PhantomReference ref = new PhantomReference<>(resource, queue);

// 单独线程消费
new Thread(() -> {
    while (!Thread.currentThread().isInterrupted()) {
        try {
            PhantomReference r = (PhantomReference) queue.remove();
            cleanup(r); // 自己写的清理方法
        } catch (InterruptedException e) {
            break;
        }
    }
}).start();

和软引用、弱引用共用 ReferenceQueue 时要注意什么

同一个 ReferenceQueue 可以注册多种引用类型,但出队时类型擦除,你拿到的是 Reference>,必须手动判断或用泛型约束。

参数差异:SoftReferenceWeakReference 构造函数也支持传 ReferenceQueue,但它们的入队时机不同:软引用在内存不足时才可能入队,弱引用在下次 GC 就入队,虚引用则在 finalize 之后(且无 finalize 方法时紧随 GC)。

  • 如果混用,建议用 instanceof 区分类型,或为不同类型分配独立队列,避免逻辑耦合
  • 虚引用入队后,其 get() 一定为 null;而软/弱引用入队前仍可能通过 get() 拿到原对象(取决于 GC 阶段)
  • 别依赖队列顺序:不同引用类型的回收时机受 GC 算法、堆配置影响,没有严格先后

容易被忽略的兼容性坑:ReferenceQueue 在 Android 和老 JDK 上的行为差异

Android 8.0(Oreo)之前,ReferenceQueue 的实现不完全遵循 JVM 规范,尤其在并发消费时可能出现漏通知;JDK 6–7 中,ReferenceQueue.remove(long) 超时逻辑有 bug,某些情况下会无限等待。

性能影响:频繁创建/注册虚引用 + 队列消费,会增加 GC 的额外负担。实测在高吞吐服务中,每秒数万次虚引用注册可能让 Reference Handler 线程成为瓶颈。

  • 生产环境慎用虚引用做高频资源管理,优先考虑 Cleaner(JDK 9+)或 AutoCloseable 显式释放
  • 若必须用,避免在循环内反复 new PhantomReferenceReferenceQueue,复用队列,引用对象可池化
  • Android 开发请确认 targetSdkVersion ≥ 26,否则虚引用行为不稳定,尤其涉及 native 资源时

真正难的从来不是怎么把引用塞进队列,而是确保队列不积压、清理不遗漏、异常不静默——这些细节不会报错,但会在某个大促凌晨三点让你翻遍 GC 日志。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

243

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

665

2024.03.01

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

409

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

586

2023.08.10

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

633

2023.08.10

页面置换算法
页面置换算法

页面置换算法是操作系统中用来决定在内存中哪些页面应该被换出以便为新的页面提供空间的算法。本专题为大家提供页面置换算法的相关文章,大家可以免费体验。

441

2023.08.14

android开发三大框架
android开发三大框架

android开发三大框架是XUtil框架、volley框架、ImageLoader框架。本专题为大家提供android开发三大框架相关的各种文章、以及下载和课程。

306

2023.08.14

android是什么系统
android是什么系统

Android是一种功能强大、灵活可定制、应用丰富、多任务处理能力强、兼容性好、网络连接能力强的操作系统。本专题为大家提供android相关的文章、下载、课程内容,供大家免费下载体验。

1784

2023.08.22

Rust异步编程与Tokio运行时实战
Rust异步编程与Tokio运行时实战

本专题聚焦 Rust 语言的异步编程模型,深入讲解 async/await 机制与 Tokio 运行时的核心原理。内容包括异步任务调度、Future 执行模型、并发安全、网络 IO 编程以及高并发场景下的性能优化。通过实战示例,帮助开发者使用 Rust 构建高性能、低延迟的后端服务与网络应用。

1

2026.02.11

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 3.5万人学习

C# 教程
C# 教程

共94课时 | 9.2万人学习

Java 教程
Java 教程

共578课时 | 63.4万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号