Java引用机制之强引用、软引用、弱引用和虚引用的实战指南

 更新时间:2025年12月12日 09:15:35   作者:拾荒的小海螺  
Java 的引用机制是垃圾回收(GC)体系中极为关键的一部分,不同引用类型代表不同的对象生命周期管理方式,Java 从 JDK 1.2 起提供了四种引用级别强引用、软引用、弱引用和虚引用,本文将逐一讲解它们的特点、内存行为、适用场景,并提供实践示例代码

1、简述

Java 的引用机制是垃圾回收(GC)体系中极为关键的一部分,不同引用类型代表不同的对象生命周期管理方式。合理使用引用类型,可以显著优化内存利用率,提高系统稳定性,尤其在缓存设计、资源管理和大规模数据处理场景中。

Java 从 JDK 1.2 起提供了四种引用级别:

  • 强引用(Strong Reference)
  • 软引用(Soft Reference)
  • 弱引用(Weak Reference)
  • 虚引用(Phantom Reference)

本文将逐一讲解它们的特点、内存行为、适用场景,并提供实践示例代码。

2、引用机制

2.1 强引用(Strong Reference)

特点

  • Java 默认的引用方式,例如:Object obj = new Object();
  • 最强引用级别,只要存在强引用,GC 永不会回收该对象。
  • 可能导致 OOM,如果强引用链无法断开。

典型应用

  • 普通对象的创建与访问。
  • 业务逻辑中绝大部分引用。

2.2 软引用(Soft Reference)

特点

  • 内存不足时,GC 会优先回收软引用指向的对象
  • 非内存不足时,不会被回收。
  • 适用于构建内存敏感缓存(如图片缓存、数据缓冲区等)。

典型应用

  • Guava Cache、部分本地缓存策略。
  • 内存有限的移动端、IoT 程序的资源管理。
import java.lang.ref.SoftReference;

public class SoftRefDemo {
    public static void main(String[] args) {
        SoftReference<byte[]> softRef = new SoftReference<>(new byte[10 * 1024 * 1024]); // 10MB
        System.out.println("Before GC: " + softRef.get());

        System.gc();
        System.out.println("After GC: " + softRef.get());  // 大概率存在

        // 占用大量内存触发回收
        try {
            byte[] bigData = new byte[100 * 1024 * 1024];
        } catch (Error e) {
            System.out.println("内存不足,软引用对象被回收: " + softRef.get());
        }
    }
}

2.3 弱引用(Weak Reference)

特点

  • 一旦 GC 执行,无论内存是否紧张,都会回收弱引用指向的对象。
  • 弱引用典型特点:朝生暮死

典型应用

  • 缓存(比软引用更“短命”)。
  • ThreadLocalMap 的 key(防止内存泄漏)。
  • 弱键 HashMap(WeakHashMap)。
import java.lang.ref.WeakReference;

public class WeakRefDemo {
    public static void main(String[] args) {
        WeakReference<String> weakRef = new WeakReference<>(new String("hello weak"));
        System.out.println("Before GC: " + weakRef.get());

        System.gc();
        System.out.println("After GC: " + weakRef.get());  // 必定为 null
    }
}

2.4 虚引用(Phantom Reference)

特点

  • 最弱的引用类型:phantomRef.get() 永远返回 null。
  • 必须结合 ReferenceQueue 使用。
  • GC 回收对象前,会将虚引用放入队列,用于资源清理。

应用场景

  • 替代 finalize(),实现更安全的资源释放。
  • 监控对象回收过程。
  • DirectByteBuffer 底层内存释放机制。
import java.lang.ref.PhantomReference;
import java.lang.ref.ReferenceQueue;

public class PhantomRefDemo {
    public static void main(String[] args) throws InterruptedException {
        ReferenceQueue<Object> queue = new ReferenceQueue<>();
        Object obj = new Object();
        PhantomReference<Object> phantomRef = new PhantomReference<>(obj, queue);

        System.out.println("Phantom get(): " + phantomRef.get()); // 永远是 null

        obj = null;
        System.gc();

        Thread.sleep(100);
        System.out.println("引用队列是否有元素: " + (queue.poll() != null));
    }
}

3、结合实际场景的应用示例

3.1 使用软引用构建简单缓存

import java.lang.ref.SoftReference;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;

public class SoftCache<K, V> {
    private final Map<K, SoftReference<V>> cache = new ConcurrentHashMap<>();

    public void put(K key, V value) {
        cache.put(key, new SoftReference<>(value));
    }

    public V get(K key) {
        SoftReference<V> ref = cache.get(key);
        return ref == null ? null : ref.get(); // 若被回收则返回 null
    }
}

3.2 WeakHashMap 自动回收 key

import java.util.WeakHashMap;

public class WeakHashMapDemo {
    public static void main(String[] args) {
        WeakHashMap<Object, String> map = new WeakHashMap<>();
        Object key = new Object();
        map.put(key, "value");

        System.out.println("Before: " + map);

        key = null;
        System.gc();

        try { Thread.sleep(100); } catch (Exception ignored) {}
        System.out.println("After: " + map);
    }
}

4、总结

  • 强引用:正常使用,生命周期最长
  • 软引用:适合可回收缓存,避免 OOM
  • 弱引用:适合避免内存泄漏,如 WeakHashMap、ThreadLocal
  • 虚引用:适用于对象回收监听,资源释放管理

掌握这四种引用方式,有助于在高性能、高并发系统中更精细地管理内存,提高系统健壮性。

引用类型GC 回收时机是否可通过 get() 获得对象常见用途
强引用永不回收可以普通对象
软引用内存不足时回收可以缓存(内存敏感)
弱引用一遇 GC 即回收可以WeakHashMap、避免内存泄漏
虚引用回收前通知不可以(always null)资源清理、监控对象生命周期

以上就是Java引用机制之强引用、软引用、弱引用和虚引用的实战指南的详细内容,更多关于Java强引用、软引用、弱引用和虚引用的资料请关注脚本之家其它相关文章!

相关文章

  • 使用concurrentHashMap如何实现缓存

    使用concurrentHashMap如何实现缓存

    文章介绍了使用ConcurrentHashMap实现缓存的线程安全性和初始化方法,以及如何处理高并发场景下的缓存清理和数据一致性问题,包括分桶、使用ConcurrentLinkedQueue以及使用CountDownLatch来确保缓存数据的不丢失
    2025-02-02
  • List集合对象中按照不同属性大小排序的实例

    List集合对象中按照不同属性大小排序的实例

    下面小编就为大家带来一篇List集合对象中按照不同属性大小排序的实例。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-03-03
  • springmvc项目使用@Valid+BindingResult遇到的问题

    springmvc项目使用@Valid+BindingResult遇到的问题

    这篇文章主要介绍了springmvc项目使用@Valid+BindingResult遇到的问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-12-12
  • MyBatis中的limit分页设置

    MyBatis中的limit分页设置

    这篇文章主要介绍了MyBatis中的limit分页设置方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2023-11-11
  • Java中的.concat()方法实例详解

    Java中的.concat()方法实例详解

    concat()方法用于将指定的字符串参数连接到字符串上,.concat()方法是一种连接两个字符串的简单方法,可以帮助我们在Java中处理字符串,对java .concat()方法用法感兴趣的朋友一起看看吧
    2024-01-01
  • httpclient staleConnectionCheckEnabled获取连接流程解析

    httpclient staleConnectionCheckEnabled获取连接流程解析

    这篇文章主要为大家介绍了httpclient staleConnectionCheckEnabled获取连接流程示例解析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-11-11
  • java发起http请求调用post与get接口的方法实例

    java发起http请求调用post与get接口的方法实例

    在实际开发过程中,我们经常需要调用对方提供的接口或测试自己写的接口是否合适,下面这篇文章主要给大家介绍了关于java发起http请求调用post与get接口的相关资料,需要的朋友可以参考下
    2022-08-08
  • Java同步锁synchronized用法的最全总结

    Java同步锁synchronized用法的最全总结

    这篇文章主要介绍了Java同步锁synchronized用法的最全总结,需要的朋友可以参考下,文章详细讲解了Java同步锁Synchronized的使用方法和需要注意的点,希望对你有所帮助
    2023-03-03
  • 云IDE:Eclipse Che:Eclipse下一代IDE(推荐)

    云IDE:Eclipse Che:Eclipse下一代IDE(推荐)

    这篇文章主要介绍了云IDE:Eclipse Che:Eclipse下一代IDE,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-09-09
  • JAVA中的Token 基于Token的身份验证实例

    JAVA中的Token 基于Token的身份验证实例

    这篇文章主要介绍了JAVA中的Token 基于Token的身份验证实例,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-08-08

最新评论