Java中的引用

news2025/1/21 12:03:00

Java中的引用

    • 强引用
    • 软引用
    • 弱引用
    • 虚引用
    • 终结器引用(FinalReference)

JDK 1.2版本之后,Java对引用的概念进行了扩充,将引用分为强引用(Strongly Reference)、软引用(Soft Reference)、弱引用(Weak Reference)和 虚引用(Phantom Reference)4种,这四种引用的强度依次逐渐减弱。 ——《深入理解Java虚拟机》

除强引用外,其他3种引用均可以在java.lang.ref包中找到它们的身影。如下图,显示了这3种引用类型对应的类,开发人员可以在应用程序中直接使用它们。

在这里插入图片描述

在Java程序中,最常见的引用类型是强引用(普通系统99%以上都是强引用),也就是我们最常见的普通对象引用,也是默认的引用类型

当在Java语言中使用new操作符创建一个新的对象,并将其赋值给一个变量的时候,这个变量就成为指向该对象的一个强引用。

强引用的对象是可触及的,垃圾收集器就永远不会回收掉被引用的对象

对于一个普通的对象,如果没有其他的引用关系,只要超过了引用的作用域或者显式地将相应(强)引用赋值为null,就是可以当做垃圾被收集了,当然具体回收时机还是要看垃圾收集策略。

相对的,软引用、弱引用和虚引用的对象是软可触及、弱可触及和虚可触及的,在一定条件下,都是可以被回收的。所以,强引用是造成Java内存泄漏的主要原因之一

强引用

强引用是最”传统“的引用的定义,指的是在程序代码中普遍存在的引用赋值,即类似 ”Object obj = new Object()“ 这种引用关系。无论任何情况下,只要强引用关系还存在,垃圾回收去就永远不会回收掉被引用的对象。 ——《深入理解Java虚拟机》

强引用具备以下特点:

  • 强引用可以直接访问目标对象。
  • 强引用所指向的对象在任何时候都不会被系统回收,虚拟机宁愿抛出oOM异常,也不会回收强引用所指向对象。
  • 强引用可能导致内存泄漏。
public class StrongReferenceTest {
    public static void main(String[] args) {
        StringBuffer str = new StringBuffer("强引用测试");
        StringBuffer str1 = str;

        str = null;  // help to gc
        System.gc(); 
        
        // 休眠3s让垃圾回收进行
        try {
            Thread.sleep(3000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        
        System.out.println(str1);
    }
}

在这里插入图片描述
可以看到,强引用关系只要存在,就不会被垃圾回收器回收。


软引用

软引用是用来描述一些还有用,但非必须的对象。只被软引用关联的对象关联着的对象,在系统将要发生内存溢出异常前,会把这些对象列入回收范围之中进行第二次回收,如果这次还没有足够的内存,才会抛出内存溢出异常。在JDK1.2之后提供了SoftReference类来实现软引用。 ——《深入理解Java虚拟机》

软引用通常用来实现内存敏感的缓存。比如:高速缓存就有用到软引用。

垃圾回收器在某个时刻绝对回收软可达的对象的时候,会清理软引用,并可选的把引用存放到一个引用队列(Reference Queue)。

类似弱引用,只不过Java虚拟机会尽量让软引用的存活时间长一些,迫不得已才清理。

// -Xmx10m -Xms10m
public class SoftReferenceTest {
    public static class User{
        private int id;
        private  String name;

        public User(int id, String name) {
            this.id = id;
            this.name = name;
        }

        @Override
        public String toString() {
            return "User{" +
                    "id=" + id +
                    ", name='" + name + '\'' +
                    '}';
        }
    }

    public static void main(String[] args) {
        // 创建对象,建立软引用
        SoftReference<User> userSoftReference = new SoftReference<User>(new User(1,"易"));
        // 等价于
//        User user = new User(1, "易");
//        SoftReference<User> userSoftReference = new SoftReference<User>(user);
//        user = null;  // 取消强引用
        // 从软引用中重新获得强引用对象
        System.out.println(userSoftReference.get());

        System.gc();
        System.out.println("After GC:");

        // 垃圾回收之后获得软引用中的对象 ,内存足够,不会回收软引用
        System.out.println(userSoftReference.get());

        try{
            // 让系统认为内存紧张
            // 设置堆空间10M,新生代大小于老年代大小默认比 1:2
            // newSize: 3.3M oldSize:6.4M
            byte[] b = new byte[1024 * 1024 * 7]; // 7M 显然放不下
        }catch(Throwable e){
            e.printStackTrace();
        }finally {
            // 再次从软引用中获取数据
            System.out.println(userSoftReference.get());
        }

    }
}

在这里插入图片描述
可以看到,当内存充足的时,GC并没有回收软引用对象,当内存不足时的隐式GC之后,软引用对象被回收了。

弱引用

弱引用也是用来描述那些非必须对象,但是它的强度比软引用更弱一些,只被弱引用关联的对象只能生存到下一次垃圾收集发生为止。当垃圾收集器开始工作,无论当前内存是否足够,都会回收掉掉只被弱引用关联的对象。在JDK1.2之后提供了WeakReference类来实现软引用。 ——《深入理解Java虚拟机》

弱引用也是用来描述那些非必需对象,只被弱引用关联的对象只能生存到下一次垃圾收集发生为止。在系统GC时,只要发现弱引用,不管系统堆空间使用是否充足,都会回收掉只被弱引用关联的对象。

但是,由于垃圾回收器的线程通常优先级很低,因此,并不一定能很快地发现持有弱引用的对象。在这种情况下,弱引用对象可以存在较长的时间

弱引用和软引用一样,在构造弱引用时,也可以指定一个引用队列,当弱引用对象被回收时,就会加入指定的引用队列,通过这个队列可以跟踪对象的回收情况。

软引用、弱引用都非常适合来保存那些可有可无的缓存数据。如果这么做,当系统内存不足时,这些缓存数据会被回收,不会导致内存溢出。而当内存资源充足时,这些缓存数据又可以存在相当长的时间,从而起到加速系统的作用。

public class WeakReferenceTest {
    public static void main(String[] args) {
        // 创建对象,建立弱引用
        WeakReference<String> weakReference = new WeakReference<>(new String("WeakReference"));

        // 从弱引用中重新获得强引用对象
        System.out.println(weakReference.get());

        System.gc();
        // 不管当前内存是否足够,都会回收它
        System.out.println("After GC");
        // 尝试从弱引用中获取对象
        System.out.println(weakReference.get());
    }
}

在这里插入图片描述
可以看到,软引用只要发生GC就会被回收。

软引用内存不足时才回收,弱引用发现即回收


虚引用

虚引用也被称为”幽灵引用“或者”幻影引用“,它是最弱的一种引用关系。一个对象是否有虚引用的存在,完全不会对其生存时间构成影响,也无法通过虚拟引用来取得一个对象实例。为一个对象设置虚引用关联的唯一目的只是为了能在这个对象被收集器回收时收到一个系统通知。在JDK1.2之后提供了PhantomReference类来实现软引用。 ——《深入理解Java虚拟机》

如果一个对象仅持有虚引用,那么它和没有引用几乎是一样的,随时都可能被垃圾回收器回收。

它不能单独使用,也无法通过虚引用来获取被引用的对象。当试图通过虚引用的get()方法取得对象时,总是null。

虚引用必须和引用队列一起使用。虚引用在创建时必须提供一个引用队列作为参数。当垃圾回收器准备回收一个对象时,如果发现它还有虚引用,就会在回收对象后,将这个虚引用加入引用队列,以通知应用程序对象的回收情况。

由于虚引用可以跟踪对象的回收时间,因此,也可以将一些资源释放操作放置在虚引用中执行和记录

public class PhantomReferenceTest {
    public  static PhantomReferenceTest obj; // 当前类对象的声明
    static ReferenceQueue<PhantomReferenceTest> phantomQueu = null; // 引用队列

    public static class CheckRefQueue extends  Thread{
        @Override
        public void run() {
            while (true) {
                if (phantomQueu != null){
                    PhantomReference<PhantomReferenceTest> objt = null;
                    try {
                        objt = (PhantomReference<PhantomReferenceTest>) phantomQueu.remove();
                    } catch (InterruptedException e){
                        e.printStackTrace();
                    }
                    if(objt != null){
                        System.out.println("追踪垃圾回收过程:PhantomReferenceTest实例被GC了");
                    }
                }
            }
        }
    }

    @Override
    protected void finalize() throws Throwable {
        // GC之前调用,有且只会被调用一次
        super.finalize();
        System.out.println("调用当前类的finalize()方法");
        // 第一次调用是对象自我拯救的最后机会
        obj = this; // 复活当前对象
    }

    public static void main(String[] args) {
        Thread t = new CheckRefQueue();
        t.setDaemon(true); // 设置为守护线程
        t.start();

        phantomQueu = new ReferenceQueue<>();
        obj = new PhantomReferenceTest();
        // 构造 PhantomReferenceTest 对象的虚引用,并指定了引用队列
        PhantomReference<PhantomReferenceTest> phantomRef = new PhantomReference<>(obj,phantomQueu);

        try {
            // 不可获取虚引用中的对象
            System.out.println(phantomRef.get());

            // 将强引用去除
            obj = null;
            //第一次GC,由于对象可复活,GC无法回收该对象
            System.gc();
            Thread.sleep(1000);
            if(obj == null){
                System.out.println("obj 是 null");
            }else {
                System.out.println("obj 可用");
            }

            System.out.println("第 2 次 GC");
            obj = null;
            System.gc(); // 一旦将obj对象回收,就会将此虚引用放到引用队列中
            Thread.sleep(1000);
            if(obj == null){
                System.out.println("obj 是 null");
            }else {
                System.out.println("obj 可用");
            }
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}

在这里插入图片描述

终结器引用(FinalReference)

Java中还有一种引用叫做终结器引用

  • 它用以实现对象的finalize ()方法,也可以称为终结器引用。
  • 无需手动编码,其内部配合引用队列使用。
  • 在cc时,终结器引用入队。由Finalizer线程通过终结器引用找到被引用对象并调用它的finalize ()方法,第二次GC时才能回收被引用对象。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/47508.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

时间序列:时间序列模型---移动平均过程(Moving Average Process)

本文是Quantitative Methods and Analysis: Pairs Trading此书的读书笔记。 我们从白噪声生成另一种时间序列。如下式&#xff1a; 这种时间序列的值由此刻的白噪声实现&#xff08;white noise realization)加上beta倍的前一刻的白噪声实现。注意这个beta跟CAPM模型的beta没有…

Linux redict 输入输出重定向 详细使用方法 文件描述符

Linux redict 重定向 Linux 重定向 在 Linux 系统中&#xff0c;我们需要输入和输出让系统与外部进行交互&#xff0c;比如在我们使用鼠标、键盘等输入设备时其实就是通过输入的方式让数据进行系统中。而系统输出一般就会打印在显示器上、刻录光盘等等。而我们要讲的重定向也…

【学习笔记70】数据劫持

一、 数据驱动视图 多次渲染页面&#xff0c;多的时候&#xff0c;比较麻烦和繁琐const box document.querySelector(.box)const obj {name: QF666,age: 18}box.innerHTML 名字: ${obj.name}; 年龄: ${obj.age};obj.age 99;box.innerHTML 名字: ${obj.name}; 年龄:…

RabbitMQ系列【16】AmqpTemplate接口详解

有道无术&#xff0c;术尚可求&#xff0c;有术无道&#xff0c;止于术。 文章目录前言AmqpTemplateAPIsendconvertAndSendreceivereceiveAndConvertreceiveAndReplysendAndReceiveconvertSendAndReceive前言 RabbitTemplate 是spring-amqp提供的一个 RabbitMQ 消息操作模板类…

【Git】rebase 和 merge 的区别

前言 今天想把本地的两个提交压缩成一个提交&#xff0c;再推送到远程。用的是rebase命令解决的&#xff0c;于是乎又捡起了之前的遗留问题&#xff1a;rebase和 merge 有什么区别&#xff1f; 用的是idea内置的git插件&#xff0c;先把idea官网对 “update project” 选择 “…

postgresql使用pg_basebackup备份与恢复

postgresql可以使用pg_dump,pg_restore等命令来进行备份与恢复&#xff0c;那种情况不用停止pgsql服务&#xff0c;只需要执行备份恢复命令即可。 今天介绍的这种备份方式&#xff0c;类似于文件系统的备份与恢复&#xff0c;它需要使用pg_basebackup命令来进行备份&#xff0c…

C#医院门诊会员管理系统源码 通用会员系统源码

C#通用医院会员管理系统源码 源码分享&#xff01; 本系统使用的技术为NhibernateEF,底层完全封装&#xff0c;可二次使用快速开发。 本技术具有以下特点&#xff1a; 1.面向对象方式访问数据库&#xff0c;摆脱SQL&#xff1b; 2.可移植性强&#xff0c;支持所有流行的数据…

光格科技递交科创板上会稿:拟募资6亿 预计年营收3亿

雷递网 雷建平 11月29日苏州光格科技股份有限公司&#xff08;简称&#xff1a;“光格科技”&#xff09;日前递交上会稿&#xff0c;准备在科创板上市。光格科技计划募资6亿&#xff0c;其中&#xff0c;3.1亿元用于分布式光纤传感系统升级研发及量产项目&#xff0c;8000万元…

Java EE|多线程基本操作

文章目录一、一个简单的线程程序及运行二、线程的创建三、线程类——Thread详解常见构造方法常见几个属性线程的启动——start()线程的中断线程的等待——join()线程引用的获取线程的休眠四、多线程编程效率举例一、一个简单的线程程序及运行 在写这样一个代码之前&#xff0c…

web框架

目录 1 左右分割窗口 2 上下分割窗口 3 嵌套分割窗口 4 内联框架 框架的作用是把浏览器窗口划分成若干个小窗口&#xff0c;每个小窗口可以分别显示不同的网页。 框架的基本结构主要分为框架集和框架两个部分&#xff0c;在网页中分别用<frameset>和<frame>标记…

Netty进阶——粘包与半包(代码示例)

目录一、消息粘包和消息半包的概述1.1、消息粘包1.2、消息半包二、粘包现象代码示例2.1、粘包现象服务端示例代码2.2、粘包现象客户端示例代码2.3、分别启动服务端&#xff0c;客户端&#xff0c;查看服务端结果输出三、半包现象代码示例3.1、半包现象服务端示例代码3.2、半包现…

【JavaSE】学习异常

前言&#xff1a; 作者简介&#xff1a;爱吃大白菜1132 人生格言:纸上得来终觉浅&#xff0c;绝知此事要躬行 如果文章知识点有错误的地方不吝赐教&#xff0c;和大家一起学习&#xff0c;一起进步&#xff01; 如果觉得博主文章还不错的话&#xff0c;希望三连支持&#xff01…

D-020 SPI FLASH硬件电路设计

SPI FLASH硬件电路设计1 简介2 EEPROM 和SPI Flash的区别3 电路设计实战4 电路设计要点1 简介 SPI FLASH(Serial Peripheral Interface)是串行外设接口的缩写&#xff0c;是一种高度、全双工、同步的通信总线。一般应用在MCU与外围设备之间通讯&#xff0c;广泛应用在FLASH&am…

从模型容量的视角看监督学习

这几天看离线强化学习瞎想的&#xff0c;不一定正确&#xff0c;仅记录个人想法 文章目录1. 监督学习的本质2. 容量视角下的模型选择、正则化和归纳偏置3. 几点启发1. 监督学习的本质 我认为监督学习的本质在于在过拟合和欠拟合之间取得平衡&#xff0c;捋一下逻辑 我们知道&a…

基于JAVA+SpringMVC+Mybatis+Vue+MYSQL的医药销售管理系统

项目介绍 药品一直以来在人类生活中扮演着非常重要的角色&#xff0c;随着时代的飞速发展&#xff0c;人们基本已经告别了那个缺医少药的年代&#xff0c;各大药房基本随处都可以购买&#xff0c;但是很多时候因为没有时间或者在药店很难找到自己想要购买的药品&#xff0c;所…

[附源码]计算机毕业设计springboot个人博客系统

项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; SSM mybatis Maven Vue 等等组成&#xff0c;B/S模式 M…

Windows远程连接centos7图形化界面,安装xrdp

Windows远程连接centos7图形化界面&#xff0c;安装xrdp写在最前面准备工作查看ubuntu系统的版本信息和gcc版本尝试进入图形化界面更新yum下载安装图形化界面查询本地是否有Server with GUI group安装Server with GUI设置系统启动默认加入GUI界面安装epel库&#xff08;epel是社…

Djiango 模版系统详解(ORM数据模型-使用mysql数据库增删改查)

djiango模版系统&#xff1a; 用于自动渲染一个文本文件&#xff0c;一般用于HTML页面&#xff0c;模版引擎渲染的最终HTML内容返回给客户端浏览器 模版系统分成两部分 静态部分&#xff1a; 例如html css .js 动态部分 djiango 模版语言&#xff0c;类似于jinja语法变量定义&…

ArcGIS QGIS学习二:图层如何只显示需要的部分几何面数据(附最新坐标边界下载全国省市区县乡镇)

文章目录前言准备SHP数据ArcMap 的筛选QGIS 的筛选如何编写查询条件前言 当我们用GIS软件打开一个SHP文件的时候&#xff0c;会显示出里面全部的几何图形&#xff0c;假如我只想要其中的一部分数据显示出来&#xff0c;其他的均不要显示&#xff0c;有那么几种操作方法。 我们…

UE4使用蓝图实现角色冲刺

又学了几天&#xff0c;前面记录了如何使用蓝图实现开关门&#xff0c;这次来实现一下角色的冲刺、瞬移的操作 一般玩游戏的时候&#xff0c;可能都会有按左shift键让角色从行走变成奔跑的状态&#xff0c;又或者双击回避键角色瞬移躲避等操作 那就先实现一下加速奔跑吧&…