在Java里强引用软引用弱引用虚引用是什么_Java引用类型解析

Java中强、软、弱、虚引用本质是四种不同松紧度的对象持有方式:强引用阻止GC回收;软引用在内存不足时才被回收,适合缓存;弱引用在每次GC时都可能被回收,用于避免泄漏;虚引用无法获取对象,仅用于回收前通知,须配合ReferenceQueue使用。

Java里强、软、弱、虚引用的本质,是四种不同“松紧度”的对象持有方式——它们不改变你写代

码的语法,但直接决定GC(垃圾回收器)会不会、什么时候、以什么优先级把对象收走。

强引用:只要它还“抓着”,GC就绝不动手

这是你每天都在用的默认行为:Object obj = new Object() 中的 obj 就是强引用。只要栈上或静态区还有变量直接指向这个对象,哪怕内存快爆了,JVM 宁愿抛 OutOfMemoryError 也不会回收它。

  • 常见错误现象:缓存 Map 里长期存着大对象(如图片、JSON),又没手动清理,最终 OOM
  • 使用场景:业务主对象、配置单例、Spring Bean 默认注入关系
  • 容易踩的坑:忘记在监听器/回调注册后解绑,导致被强引用链锁死,引发内存泄漏

软引用:内存告急时才放手,适合做缓存

SoftReference cacheRef = new SoftReference(new byte[1024 * 1024 * 5]) —— 这个引用不会阻止 GC,但 GC 会“手下留情”,只在真正要 OOM 前才把它干掉。

  • 关键机制:JVM 会按“最近最少使用”+“内存压力”综合判断是否回收,不是简单看引用类型
  • 实操建议:不要依赖 softRef.get() 必然返回非 null;每次取值前必须判空,且做好重建逻辑
  • 性能影响:频繁创建/释放软引用对象会增加 GC 扫描开销;高并发下建议配合 ReferenceQueue 异步清理,避免阻塞主线程

弱引用:GC 一来就消失,适合临时绑定和自动清理

WeakReference listenerRef = new WeakReference(new MyListener()) —— 只要一次 GC 触发,不管内存多富裕,这个引用指向的对象大概率就没了。

  • 典型应用:WeakHashMap 的 key 就是弱引用;注册监听器时用弱引用可避免 Activity/Fragment 泄漏
  • 注意点:不能用于需要稳定生命周期的场景;get() 返回 null 是常态,不是异常
  • 陷阱:别在构造弱引用后还保留强引用(比如同时把对象赋给另一个变量),否则弱引用形同虚设

虚引用:拿不到对象,只等“死亡通知”

PhantomReference phantRef = new PhantomReference(obj, queue) —— 调用 phantRef.get() 永远返回 null,唯一作用是让对象在被回收前“路过”一次 ReferenceQueue

  • 为什么不用 finalize()?因为虚引用不阻塞 GC,而 finalize() 已废弃且严重拖慢回收速度
  • 实操必需项:必须搭配 ReferenceQueue 使用;需另起线程轮询队列,执行资源释放(如关闭文件句柄、释放堆外内存)
  • 容易忽略的细节:虚引用对象进入队列 ≠ 已回收,只是“已标记为待回收”;真正的内存释放发生在后续 GC 阶段

真正难的不是记住四类引用的定义,而是判断:当前对象该不该被“保命”、能活多久、谁负责善后。强引用是默认保险带,软/弱是弹性缓冲,虚引用则是临终遗嘱执行人——选错类型,轻则缓存失效,重则内存雪崩。