简述Java中的四种引用类型

 更新时间:2021年04月13日 11:50:19   作者:布禾  
从JDK1.2版本开始,把对象的引用分为四种级别,从而使程序能更加灵活的控制对象的生命周期。这四种级别由高到低依次为:强引用、软引用、弱引用和虚引用,下面分别介绍下这四种引用。

简介

从JDK1.2版本开始,把对象的引用分为四种级别,从而使程序能更加灵活的控制对象的生命周期。这四种级别由高到低依次为:强引用、软引用、弱引用和虚引用,下面分别介绍下这四种引用。

强引用

强引用是最常用的引用类型,如下所示,new Object()会创建一个Object对象并存储在堆上,变量object存储对该对象的强引用。

Object object = new Object();

强引用不会被垃圾回收,所以要想回收该对象,则应该将指向该对象的变量显示设为null,这样该对象就由强引用转变为无引用了。

示例:

public class ReferenceDemo {
    public static void main(String[] args) throws IOException {
        //强引用不会被垃圾回收
        ReferenceDemo referenceDemo = new ReferenceDemo();
        //强引用转变为无引用,无引用可被垃圾回收
        referenceDemo = null;
        //触发垃圾回收
        System.gc();
        //阻塞主线程,等待垃圾回收线程执行
        System.in.read();
    }

    //对象被回收之前调用
    @Override
    protected void finalize() throws Throwable {
        super.finalize();
        System.out.println("-----finalize-----");
    }
}

软引用

软引用是使用SoftReference创建的,在内存空间充足的情况下,软引用不会被回收,而在内存空间不足虚拟机抛出OutOfMemoryError之前,软引用将会被回收。

示例:

public class ReferenceDemo {
    public static void main(String[] args) throws InterruptedException {
        //创建ReferenceDemo对象的软引用
        SoftReference<ReferenceDemo> softReference = new SoftReference<>(new ReferenceDemo());
        //触发垃圾回收
        System.gc();
        //阻塞主线程,等待垃圾回收线程执行
        Thread.sleep(5000);
        //softReference.get()返回软引用对象,如果对象已经被垃圾回收,则返回null
        System.out.println(softReference.get());

        //创建25M的字节数组
        byte[] bytes = new byte[1024 * 1024 * 25];
        //内存已经不足,阻塞主线程,等待垃圾回收线程执行
        Thread.sleep(5000);
        //重新输出软引用对象
        System.out.println(softReference.get());
    }
}

/*
 * 输出结果:
 * com.buhe.demo.demos.reference.ReferenceDemo@76fb509a
 * null
 */

注意:示例运行前需要设置堆内存大小为30M(-Xmx30m -Xms30m)。

用途:软引用可以用于对内存空间敏感的缓存,缓存的对象一直保存,直到内存空间不足而被回收。

弱引用

弱引用是使用WeakReference创建的,在垃圾回收线程执行过程中,只要找到了弱引用,不管内存空间是否足够,弱引用对象都将被回收。由于垃圾回收线程是一个优先级很低的线程,因此不一定会很快发现那些只具有弱引用的对象。

示例:

public class ReferenceDemo {
    public static void main(String[] args) throws InterruptedException {
        //创建ReferenceDemo的弱引用
        WeakReference<ReferenceDemo> weakReference = new WeakReference<>(new ReferenceDemo());
        //weakReference.get()返回弱引用对象,如果对象已经被垃圾回收,则返回null
        System.out.println(weakReference.get());

        //触发垃圾回收
        System.gc();
        //阻塞主线程,等待垃圾回收线程执行
        Thread.sleep(3000);

        //重新输出弱引用对象
        System.out.println(weakReference.get());
    }
}

/*
 * 输出结果:
 * com.buhe.demo.demos.reference.ReferenceDemo@76fb509a
 * null
 */

用途:弱引用也可以用于缓存,可以参考WeakHashMap类。

虚引用

虚引用是使用PhantomReference创建的,它是所以引用类型中最弱的。虚引用对象和没有引用的对象相同,可以在任何时候被垃圾回收,并且虚引用必须要与引用队列一起使用。

当垃圾回收线程回收一个虚引用对象时,它将在垃圾回收后销毁该对象,并将PhantomReference添加到引用队列中。

示例:

public class ReferenceDemo {
    public static void main(String[] args) throws InterruptedException {
        //创建引用队列
        ReferenceQueue<Object> referenceQueue = new ReferenceQueue();
        //创建ReferenceDemo的虚引用
        PhantomReference<ReferenceDemo> phantomReference = new PhantomReference<>(new ReferenceDemo(), referenceQueue);
        //phantomReference.get()总是返回null
        System.out.println("phantomReference.get():" + phantomReference.get());
        //轮询此队列,查看是否有可用的Reference对象,有则返回该对象,否则返回null
        System.out.println("referenceQueue.poll():" + referenceQueue.poll());

        //触发垃圾回收
        System.gc();
        //阻塞主线程,等待垃圾回收线程执行
        Thread.sleep(3000);
        System.out.println("------垃圾回收之后------");

        System.out.println("phantomReference.get():" + phantomReference.get());
        System.out.println("referenceQueue.poll():" + referenceQueue.poll());
    }
}

/*
 * 输出结果:
 * phantomReference.get():null
 * referenceQueue.poll():null
 * ------垃圾回收之后------
 * phantomReference.get():null
 * referenceQueue.poll():java.lang.ref.PhantomReference@76fb509a
 */

用途:虚引用可以用于精确的检测对象何时从内存中删除,通过检查引用队列来判断对象是否已经被回收。

以上就是简述Java中的四种引用类型的详细内容,更多关于Java 引用类型的资料请关注脚本之家其它相关文章!

相关文章

  • 利用SpringBoot和LiteFlow解锁复杂流程

    利用SpringBoot和LiteFlow解锁复杂流程

    随着业务的复杂化,企业需要更加高效、便捷地管理自己的业务流程,这就需要借助一些流程引擎实现,今天,我们就来介绍一种基于Java语言开发的轻量级工作流引擎——LiteFlow,以及如何在Spring Boot框架中集成它,从而提高企业的工作效率和开发效率
    2023-06-06
  • Springboot接收Get参数实践过程

    Springboot接收Get参数实践过程

    本文主要介绍了在Spring Boot中如何接收不同类型的请求参数,包括在路径中直接传递参数、跟在问号后面传递参数、使用Map接收参数、接收数组以及使用对象接收参数等方法
    2024-12-12
  • IDEA如何对单个的java class文件打成jar包

    IDEA如何对单个的java class文件打成jar包

    这篇文章主要介绍了IDEA如何对单个的java class文件打成jar包问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2023-08-08
  • java根据方法名称取得反射方法的参数类型示例

    java根据方法名称取得反射方法的参数类型示例

    利用java反射原理调用方法时,常先需要传入方法参数数组才能取得方法。该方法参数数组采用动态取得的方式比较合适
    2014-02-02
  • 深入理解Java设计模式之策略模式

    深入理解Java设计模式之策略模式

    这篇文章主要介绍了JAVA设计模式之策略模式的的相关资料,文中示例代码非常详细,供大家参考和学习,感兴趣的朋友可以了解下
    2021-11-11
  • Java黑科技之通过Google Java Style 文件配置IDEA和Ecplise代码风格

    Java黑科技之通过Google Java Style 文件配置IDEA和Ecplise代码风格

    在日常开发中,多人团队协作开发一个项目是很常见的,特别是大公司,这就会涉及到多人在一个工程上开发代码.无规矩不成方圆,一个好的代码风格,更加有利于团队协作,减少代码冲突,提高代码可阅读性,美观性.本文就带着大家仔细研究这个黑科技 ,需要的朋友可以参考下
    2021-05-05
  • java序列化的种类和使用场景详解

    java序列化的种类和使用场景详解

    本文详细介绍了序列化的概念、Java内置序列化、自定义序列化、第三方序列化框架(如Kryo、Protobuf)以及在分布式系统和RPC框架中的应用,通过比较不同序列化方式的优缺点,指导开发者选择合适的序列化方案,以确保系统的性能、安全性和可维护性
    2025-01-01
  • Spring框架中一个有用的小组件之Spring Retry组件详解

    Spring框架中一个有用的小组件之Spring Retry组件详解

    Spring Retry 是从 Spring batch 中独立出来的一个功能,主要实现了重试和熔断,对于那些重试后不会改变结果,毫无意义的操作,不建议使用重试,今天通过本文给大家介绍Spring Retry组件详解,感兴趣的朋友一起看看吧
    2021-07-07
  • springboot自动装配之@ComponentScan使用方式

    springboot自动装配之@ComponentScan使用方式

    @componentScan注解用于扫描指定路径下的组件,并自动将它们注册为Spring Bean,该注解支持多种过滤规则,可以自定义扫描过滤规则,Spring Boot通过ConfigurationClassPostProcessor处理@ComponentScan注解,并在启动时创建和注册BeanDefinition对象
    2025-01-01
  • Java面试题冲刺第八天--Spring框架2

    Java面试题冲刺第八天--Spring框架2

    这篇文章主要为大家分享了最有价值的三道Spring框架面试题,涵盖内容全面,包括数据结构和算法相关的题目、经典面试编程题等,感兴趣的小伙伴们可以参考一下
    2021-07-07

最新评论