消除过期的对象引用的理解

2023-11-04

什么是过期的对象引用?

我们通过简单的栈实现来引入过期的对象引用。

public class Stack {
    private Object[] elements;
    private int size = 0;
    private static final int DEFAULT_INITIAL_CAPACITY = 16;

    public Stack(){
        elements = new Object[DEFAULT_INITIAL_CAPACITY];
    }

    public void push(Object e){
        ensureCapacity();
        elements[size ++] = e;
    }

    public Object pop(){
        if (size == 0){
            throw new EmptyStackException();
        }
        return elements[--size];
    }

    private void ensureCapacity(){
        if (elements.length == size){
            elements = Arrays.copyOf(elements, 2 * size + 1);
        }
    }

}

实际上,这段程序中并没有很明显的错误。无论如何测试,它都会成功地运行通过每一项测试,但这个程序中隐藏着一个问题。不严格地讲,这段程序有一个”内存泄漏“, 随着垃圾回收器活动的增加,或者由于内存占用的不断增加,程序性能的降低会逐渐表现出来。在极端的情况下,这种内存泄露会导致磁盘交换,甚至程序失败,但这种情况比较少见。

那么,程序中哪里发生了内存泄漏呢?实际上,当栈先增长后收缩时,即使我们执行push后再pop,这时候弹出来的对象将不会被当作垃圾回收,即便使用栈的程序不会再引用这些对象,也不会被回收。因为栈内部维护着这些对象的过期引用。

过期引用:指的是永远不会再被解除的引用。

在我们的stack例子中,凡是在elements数组的”活动范围“之外的任何引用都是过期的,这里的活动部分指的是elements中下标小于size的那些元素。

如何解决stack中的过期引用问题?
实际上,一旦对象引用已经过期,只需清空这些引用即可。

public Object pop(){
        if (size == 0){
            throw new EmptyStackException();
        }
        Object result = elements[--size];
        //清空引用
        elements[size] = null;
        return result;
    }

那么可不可以理解为,当对象引用过期了,直接设置为null即可。
我们再看一个例子:

List<String> list = new ArrayList<>();
String str = "java";
list.add(str);
str  = null;

那么通过上面的方法,创建str的内存空间是否就被回收了呢?实际上,并没有,list依然持有对str的引用,所以创建str时的内存空间是不会被回收的。对于这种引用,我们称为”无意识的内存引用“

为了更好地解决”无意识的内存引用“问题,我们先要明白对象间相互依赖是怎么样的?

Object object1 = new Object();
Object object2 = object1;
Object object3 = object2;

其对应的引用图如下:


Paste_Image.png


因此,如果我们要回收Object1所创建的内存空间的话,单纯地设置object1 = null是无法回收的,因为object2和object3都保留着对内存的引用。

因此,要消除 ”无意识的引用“ 必须删除掉所有对内存空间的引用。

那么对于之前的list的例子,我们可以通过list.remove(str)或者list = null方法来删除对空间的引用,从而使得垃圾回收器可以回收到该内存空间。

为了更好地解决问题,我们要先弄清楚两个关键字:Java的垃圾回收机制 Java的内存泄漏

Java的垃圾回收机制:

Java中的对象是在堆中分配,对象的创建有2中方式:new或者反射。对象的回收是通过垃圾收集器,JVM的垃圾收集器简化了程序员的工作,但是却加重了JVM的工作,这是Java程序运行稍慢的原因之一,因为GC为了能正确释放对象,必须监控每一个对象的运行状态,包括对象的申请、引用、被引用、赋值等,GC都要进行监控,监控对象的状态是为了更加准确、及时地释放对象,而释放对象的根本原则就是该对象不再被引用。


Java的内存泄漏:

  • 什么是内存泄漏?
    对象已经没有被应用程序使用,但是垃圾回收器没办法移除它们,因为还在被引用着。

  • 为何会产生内存泄漏?
    Java有垃圾回收机制,那么还存在内存泄露吗?答案是肯定的,所谓的垃圾回收GC会自动管理内存的回收,而不需要程序员每次都手动释放内存,但是如果存在大量的临时对象在不需要使用时并没有取消对它们的引用,就会吞噬掉大量的内存,很快就会造成内存溢出。

内存泄漏的几种情形:

a. 类是自己管理内存的,比如我们上面所举的stack的例子,对于这种情况,一旦元素被释放掉了,则该元素中包含的任何对象引用都应该被清空。

b. 内存泄漏的另一个常见来源是缓存。一旦你把对象引用放到缓存中,它就很容易被遗忘掉,从而使得它不再有用之后很长一段时间内仍然留在缓存中。解决方法是使用WeakHashMap

c. 内存泄漏的第三个常见来源是监听器和其他回调。如果你实现了一个API,客户端在这个API中注册调用,却没有显式地取消注册,那么除非你采取某些措施,否则它们就会积聚。解决方法为确保回调立即当作垃圾回收的最佳方法是只保存它们的弱引用。



作者:大海孤了岛
链接:http://www.jianshu.com/p/65d413287090





Java的垃圾回收机制并不代表我们不需要考虑内存管理的问题。

 

考虑:

public class Stack {
    pprivate Object[] elements;
    private int size = 0;
    private static final int DEFAULT_INITAL_CAPACITY = 16;
    
    public Stack() {
        elements = new Object[DEFAULT_INITAL_CAPACITY];
    }
    
    public void push(Object e) {
        ensureCapacity();
        elements[size++] = e;
    }
    
    public Object pop() {
        if(size == 0) {
            throw new EmptyStackException();
        }
        return elements[--size];
    }
    
    private void ensureCapacity() {
        if(elements.length == size)
            elements = Arrays.copyOf(elements, 2 * size + 1);
    }
}

这是自己编写的一个栈。

这段程序没有任何明显的错误,但这个程序中隐藏着一个问题,内存泄漏。

如果一个栈先是增长,然后收缩,那么从栈中弹出来的对象不会被当作垃圾回收,这是因为栈内部仍然维护着这些过期对象的引用,所谓过期引用是指elements中下标大于等于size的那些元素(由于栈会增长收缩,所以这是完全有可能的),如果一个对象被无意识地(即我们不希望地)保留下来,那么这个引用所引用的其他对象也不会被垃圾回收机制回收,因为仍然存在着引用,GC认为这些对象仍然是会被使用的。随着程序的运行时间增长,过期的对象引用占用的内存会越来越大,导致程序性能下降。

 

这类问题的解决方法是:一旦对象引用已经过期,只需清空这些引用即可。

public Object pop() {
        if(size == 0) {
            throw new EmptyStackException();
        }
        Object result = elements[--size];
        elements[size] = null;//显式地清空引用
        return result;
}

清空对象引用应该是一种例外,而不是一种规范行为,不要求也不建议程序员对于每个对象引用,一旦程序不再用,就把它清空,这通常会把代码弄的很混乱。

消除过期引用的最好办法是在最紧凑的作用域范围内定义每一个变量,当作用域被执行完,GC会自动把过期作用域的变量回收掉。

 

什么时候应该自己清空引用?

一般而言,只要是自己管理内存,就应该警惕内存泄漏问题。假如你开辟了一段内存空间,并一直持有这段空间的引用,就有责任管理它,因为GC无法自动完成对你承诺管理的内存的回收,除非你告诉它(显式地清空引用)。

 

在JDK中,已经有现成的Stack类供我们使用,来看看它是怎么实现pop的:

public synchronized E pop() {
        E       obj;
        int     len = size();

        obj = peek();//这个方法将栈顶的元素取出来,但并不会把栈顶元素弹出
        removeElementAt(len - 1);//这是Stack父类的方法,将栈顶元素弹出

        return obj;
}

public synchronized void removeElementAt(int index) {
        modCount++;
        if (index >= elementCount) {
            throw new ArrayIndexOutOfBoundsException(index + " >= " +
                                                     elementCount);
        }
        else if (index < 0) {
            throw new ArrayIndexOutOfBoundsException(index);
        }
        int j = elementCount - index - 1;
        if (j > 0) {
            System.arraycopy(elementData, index + 1, elementData, index, j);
        }
        elementCount--;
        elementData[elementCount] = null; /* to let gc do its work *//可以看到,jdk的实现就是显式地把引用清空,以此告诉GC将过期引用回收
}

 

内存泄泄漏通常不会表现成明显的失败,可以在系统中存在很多年,只有通过检查代码,或借助Heap剖析工具才能发现内存泄漏问题。所以要尽量在内存泄漏发生之前就知道如何预测此类问题。


本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

消除过期的对象引用的理解 的相关文章

  • maven项目引入外部jar包,并实现打包

    1 在项目模块的src目录下创建 lib 文件 将要引用的jar包放到lib下 2 在该项目模块内的pom xml内引入依赖
  • 向svn服务器添加新项目

    确保电脑上已经安装了TortoiseSVN客户端 在之前已经从svn上checkout代码下来的文件夹中 点击右键 TortoiseSVN gt Repo browser 这样就可以看到整个svn服务器的目录了 得到svn服务器目录后 就可
  • ZooKeeper实战面试题(2021.11)

    文章目录 ZooKeeper是什么 ZooKeeper 提供了什么 Zookeeper 文件系统 Zookeeper 怎么保证主从节点的状态同步 恢复模式 广播模式 四种类型的数据节点 Znode Zookeeper Watcher 机制
  • 关于 vue 做文件导出的总结

    关于文件导出有2中方法 一种是后端写接口 前端调用 一种是纯前端做 下面分别介绍这两种方法 一 后端做下载 导出功能 前端调接口 注意 需要指定服务器响应的数据类型 gt responseType blob 导出 下载 import req
  • Java常用对象API——Map集合

    java util 接口 Map
  • Kettle的表或视图不存在问题【已解决】

    1 问题描述 在用Kettle做job的时候 报如下的错 2019 11 18 14 28 42 OUT FICP PARAM DATA 2 0 ERROR version 8 3 0 0 371 build 8 3 0 0 371 fro
  • Ubuntu常用终端命令

    Ubuntu常用终端命令 1 显示任务管理器 ps aux 2 kill进程 kill PID号 3 后台运行程序 nohup python3 xxxx py 4 查看文件列表 ls 5 进入文件夹 cd 文件夹名 6 解压与压缩命令 6
  • 数字水印-期末复习

    期末复习时一边复习 自学 一边记录所得 有点儿乱但能明白数字水印是个啥的话对着这个复习还是比较有用 RSA 可用于加密 数字签名 密钥分配 PGP PKI等 对RSA的主要支持和批评 形式简单 易于理解 研究深入 支持广泛 既能用来加密 又
  • 静态vector容器成员变量的定义和初始化

    想要定义一个静态容器成员变量 保存数据以便后面共享 1 要现在 h文件的类内先声明该成员 class A public static const int vecSize COMM NUMBERS 整形静态常量可以直接初始化 static v
  • 解决windows10资源管理器无限刷新、高占用的解决方法

    首先是我使用Edge添加pdf笔记的时候 Edge浏览器自动崩溃 我就重新打开 于是就开始了疯狂的无限刷新 并且cpu高占用 如果不是因为这个原因导致的无限刷新 那么这个方法或许不适合 这是一个关于浏览器读取pdf的一个bug 具体解决方案
  • 【Node.js安装与配置(详细步骤)及vue项目配置】

    Node js安装与配置 详细步骤 前言 下载Node js 安装Node js 添加环境变量 新建 node global 和 node cache 两个文件夹 查看是否配置成功 可能由于权限问题导致不成功 需要设置文件权限 vue项目配
  • 取消请求、axios实现abort

    缘由 工作项目使用fetch 暂无法提供abort 取消请求 功能 虽然官方说可以使用 AbortSignal对象的实例 将允许通过AbortController与fetch请求通信或者终止fetch 稍微复杂了也不好封装 于是将目光转回a
  • 香港服务器部署网站慢,用香港云主机服务器网站慢怎么解决?

    用香港云主机服务器搭建的网站出现了卡顿慢的情况 要怎么解决呢 这是服务器的问题吗 服务器之家来为各位用户进行一个简要的分析 希望对大家有帮助 1 检查本地客户端 本地客户端访问网络诊断分析系统 测试本地访问各域名的速度 根据测试结果 确认本
  • c++学习:1.变量定义

    1 列表初始化 c 语言定义了初始化的好几种形式 例如 int a 0 int a 0 int a 0 int a 0 使用花括号初始化是c 11标准 当用于内置类型的变量时 该种初始化 花括号 有一个重要特点 如果我们使用列表初始化且初始
  • php设置表格宽度,php-如何使用preg_replace将表格宽度更改为100%

    我想将表格宽度更改为100 如果宽度值以像素为单位 我的意思是 如果表格宽度看起来像width 500 或width 500px 那么我想用100 替换它 我的意思是像这个width 100 有人可以帮我preg replace吗 cont
  • vue文本点击样式设置

    vue文本点击样式设置 嘚吧嘚 干就完了 光标边小手 文本域样式修改 hover语法 语法一 语法二 语法三 语法四 学以致用 效果实现 嘚吧嘚 相信当家在写代码的过程中 文本的点击事件是常有的吧 如历史搜索记录 页面跳转等 本次就就分享一
  • win11 vmware 显示 “未能启动虚拟机“ 解决办法

    方法1 我使用这个方法解决 1 按下 win s 键 或直接搜索 系统信息 并打开 2 向下找到 基于虚拟化的安全性 如果是 正在运行 使用下面的方法将它关闭 3 搜索 内核隔离 打开窗口后将内存完整性关闭 如果是关闭状态 直接进行下一步
  • C++stack容器

    1 stack 基本概念 概念 stack是一种先进后出 First In Last Out FILO 的数据结构 它只有一个出口 栈中只有顶端的元素才可以被外界使用 因此栈不允许有遍历行为 栈中进入数据称为 入栈 push 栈中弹出数据称
  • BES提示音

    提示音修改介绍 Hi 大家好 大家都知道在开发TWS耳机或者立体声耳机时客户都会自定义提示音 所以现在每个平台都有开放自定义提示音的功能 如 高通 络达 中科蓝讯 炬芯等 下面咱们就讲解下BES添加提示音的过程 在添加自定义提示音之前需要获

随机推荐

  • Qualcomm Camera基础

    高通将android的camera模块重新修改了一下 与原生的方式存在一些差异 这里将前段时间学习的一些零散知识进行一下总结 便于以后查阅 1 整个模块主要巡行三个主线程 control config及frame control用来执行总的
  • linux多用户密钥登陆,多用户,多(种\个)密钥,SSH 密钥登录linux服务器

    接上文Linux服务器采用密钥认证登录 多用户 多 种 个 密钥 SSH 密钥登录linux服务器 多用户 多种密钥算法 rsa dsa SSH 私钥登录linux Red Hat CentOS Fedora Debian Ubuntu 服
  • Linux指令——crontab

    crontab指令的作用是周期性的自动执行文件 目录 一 安装 二 使用 一 编辑指令 第一步进入crontab编辑页面 第二步输入crontab指令 二 删除指令 三 拓展 比如我需要每天晚上7点执行一个文件 那么就可以使用crontab
  • 算法入门之基本数据结构:链表

    前面我们简单的对队列和栈有了个了解 今天我们还要将一种数据结构 Java中很多集合类都是由这几种数据结构演变而来的 除了队列和栈还有我们今天要说的链表 链表其实还是蛮复杂的 在C中有个指针用来实现 很多人说java不存在指针概念 是不是就不
  • 【树莓派】配置树莓派,实现文件传输

    安装操作系统 选择的是官方提供的Raspbian 进到后选择 RASPBIAN STRETCH WITH DESKTOP是图形界面 RASPBIAN STRETCH LITE是命令行界面 一般下载RASPBIAN STRETCH LITE即
  • CVE-2022-40684 Fortinet(飞塔)身份验证绕过漏洞

    GitHub地址 https github com hughink CVE 2022 40684 漏洞简介 Fortinet 飞塔 是一家全球知名的网络安全产品和安全解决方案提供商 其产品包括防火墙 防病毒软件 入侵防御系统和终端安全组件等
  • STM32实现IAP功能的学习笔记

    STM32实现IAP功能的学习笔记 最近因项目需求要实现STM32的在线升级即IAP功能 先将这几天的学习体会和IAP的具体实现总结出来 分享给大家 希望对同样实现IAP的童鞋有所帮助 文中最后会上传名为STM32 Update zip的压
  • 解释React为什么onClick函数绑定都用箭头函数()=>{}

    前言 起因在于今日同事问起 发现不能很好的解释这个问题 所以自己下来重新屡了一下思路 记录一下 一 什么是箭头函数 以及为什么React里要用箭头函数 1 下面这两个函数相同 只是react组件化 不支持在组建内部写function 所以现
  • [Xilinx FPGA] #8 Xilinx Power Estimator[XPE, 功耗估计器]的使用方法

    对于 FPGA 设计来说 设计结果的功耗是较为重要的一个设计指标 有时在设计完成前对设计的功耗有一个大体的估计 Xilinx 专门为此设计了一个工具 以使设计者可以在设计完成前根据预设对功耗进行大致的预估 可参考 Xilinx Power
  • 服务因 找不到指定的模块。 服务特定错误而停止。

    新装了sql server 2008 发现sqlserver 服务没法起来 查看系统日志是7024如下 c sharp view plain copy 事件类型 错误 事件来源 Service Control Manager 事件种类 无
  • vue3 ref属性

    vue3 ref属性 操作单个DOM元素 操作单个DOM或者组件的流程 定义一个响应式变量 把变量返回给模板使用 模板中绑定上述返回的数据 可以通过info变量操作DOM或者组件实例
  • 求绝对值函数abs()到底应该包含哪个头文件

    C C 语言的标准库里包含求绝对值的函数abs 可是它到底是在哪个头文件里面声明的呢 C语言中 求整数的绝对值abs 和labs 应该包含stdlib h 求浮点数的绝对值fabs 应该包含math h 在C 中 只需要包括cmath即可
  • 半导体材料概述 -测试测量制造

    前言 半导体概述 半导体材料作为半导体产业链上游的重要环节 在芯片的生产制造过程中起到关键性作用 根据芯片制造过程划分 半导体材料主要分为基体材料 制造材料和封装材料 其中 基体材料主要用来制造硅晶圆或化合物半导体 制造材料主要是将硅晶圆或
  • CSAPP Datalab实验

    实 验 二 题 目 DataLab 数据表示 专 业 计算机科学与技术 计算机科学与技术学院 目 录 第1章 实验基本信息 4 1 1 实验目的 4 1 2 实验环境与工具 4 1 2 1 硬件环境 4 1 2 2 软件环境 4 1 2 3
  • 量化交易/基金爬虫(一)

    先说说这个python脚本的初衷 就是为了给将来的量化交易 实现自动化亏钱做一个基石 打一下基础 但是没想到的是 在写完这个脚本之后 仅仅两天时间 我全部的收益就被赔光 哭了 import requests from bs4 import
  • 数据库原理与应用第三版何玉洁第三章课后题答案

    1 关系模型的结构 关系操作 关系的完整性约束 2 1 笛卡尔积实际上就是一个二维表 2 主键也称为主关键字 时表中的属性或属性组 用于唯一的确定一个元组 3 候选键 一个属性或属性集能够唯一表示一个关系的元组而又不包括多余的属性 则改属性
  • Unicode字符编码表

    十进制 十六进制 字符数 编码分类 中文 编码分类 英文 起始 终止 起始 终止 个 0 127 0000 007F 128 C0控制符及基本拉丁文 C0 Control and Basic Latin 128 255 0080 00FF
  • 计算机网络属性设置方法,电脑网络连接属性怎么设置

    方法1在安装系统时对Internet连接进行设置在安装WindowsXP过程中 进行Internet连接设置时 可以按提示输入用户名和密码并设置成开机自动连接宽带 这样在安装完成后每次开机就能自动连接宽带了 方法2将宽带连接添加到启动组首先
  • vue导出excel表格vue-json-excel(最新超容易详细教程)

    最近写一个报名的小程序后台 需要导出页面表格为excel 实现这个功能所以记录一下 目前大多数采取的都是 xlsx 我这里使用的是vue json excel 相对来说更容易上手实现效果 安装依赖 npm install vue json
  • 消除过期的对象引用的理解

    什么是过期的对象引用 我们通过简单的栈实现来引入过期的对象引用 public class Stack private Object elements private int size 0 private static final int D