Java虚拟机(JVM):引用计数算法

news2025/1/21 16:23:07

一、引言

我们学习了Java内存运行时区域的各个部分,其中程序计数器、虚拟机栈、本地方法栈3个区域随线程而生,随线程而灭。栈中的栈帧随着方法的进入和退出而有条不紊地执行着出栈和入栈操作。每一个栈帧中分配多少内存基本上是在类结构确定下来就已知的,因此这几个区域的内存分配和回收都具备确定性,在这几个区域内就不需要过多考虑如何回收的问题,当方法结束或者线程结束时,内存自然就跟随着回收了。

而Java堆和方法区这两个区域则有着很显著的不确定性:一个接口的多个实现类需要的内存可能会不一样,一个方法所执行的不同条件分支所需要的内存也可能不一样,只有处于运行期间,我们才能知道程序究竟会创建哪些对象,创建多少个对象,这部分内存的分配和回收是动态的。垃圾收集器所关注的正是这部分内存该如何管理,一般讨论“内存”分配与回收也仅仅特指这一部分。

二、引用计数算法

引用计数算法是一种内存管理算法,用于追踪对象的引用数量。它的基本原理是为每个对象维护一个计数器,记录当前有多少个指针指向该对象。当计数器的值变为0时,表示该对象不再被引用,可以被回收。

引用计数算法的实现思路如下:

  1. 在对象中添加一个引用计数器,初始值为0。
  2. 当有一个指针指向该对象时,引用计数器加1。
  3. 当一个指针不再指向该对象时,引用计数器减1。
  4. 当引用计数器的值为0时,表示没有指针指向该对象,可以将该对象回收。

引用计数算法的优点:

  1. 实时性:引用计数算法可以实时地进行内存回收,不需要等待垃圾回收器的运行。
  2. 简单高效:引用计数算法的实现相对简单,不需要遍历整个对象图,只需要维护计数器即可。

引用计数算法的缺点:

  1. 循环引用问题:当存在循环引用时,引用计数算法无法正确地回收内存。例如,对象A和对象B相互引用,它们的引用计数器都不会变为0,导致内存泄漏。
  2. 计数器更新开销:每次引用发生变化时,都需要更新计数器,导致额外的开销。

因为引用计数算法存在循环引用问题,所以现代的垃圾回收器往往不使用纯粹的引用计数算法,而是采用其他算法(如标记-清除算法、复制算法、标记-整理算法等)与引用计数算法结合,来解决循环引用的回收问题。

三、代码分析

以下是一个简单的引用计数算法的代码案例:

class ReferenceCounting {
    private int count; // 引用计数器
    public ReferenceCounting() {
        count = 0;
    }
    public void addReference() {
        count++;
    }
    public void removeReference() {
        count--;
    }
    public int getCount() {
        return count;
    }
}
class Object {
    private ReferenceCounting refCount; // 引用计数对象
    public Object() {
        refCount = new ReferenceCounting();
        refCount.addReference(); // 对象创建时增加引用计数
    }
    public void addReference() {
        refCount.addReference();
    }
    public void removeReference() {
        refCount.removeReference();
        if (refCount.getCount() == 0) {
            // 引用计数为0时执行回收操作
            System.out.println("Object is reclaimed.");
            // 执行回收操作
        }
    }
}
public class ReferenceCountingDemo {
    public static void main(String[] args) {
        Object obj1 = new Object(); // 创建对象1
        Object obj2 = new Object(); // 创建对象2
        obj1.addReference(); // obj1引用计数加1
        obj1.addReference(); // obj1引用计数加1
        obj2.addReference(); // obj2引用计数加1
        obj1.removeReference(); // obj1引用计数减1
        obj1.removeReference(); // obj1引用计数减1,计数为0,执行回收操作
        obj2.removeReference(); // obj2引用计数减1,计数不为0,不执行回收操作
    }
}

在上述代码中,ReferenceCounting类是引用计数器类,用于记录对象被引用的次数。Object类是被引用的对象类,其中包含了一个ReferenceCounting对象。当创建对象时,引用计数加1,当移除对象引用时,引用计数减1。当引用计数为0时,表示对象不再被引用,可以执行回收操作。 在ReferenceCountingDemo类的main方法中,我们创建了两个对象obj1obj2,分别增加和减少引用计数,演示了引用计数算法的基本原理。

在下一个案例前,我们首先要学会在IDEA中输出gc日志信息:

循环引用代码分析:

class A {
    private B b;
    public void setB(B b) {
        this.b = b;
    }
}
class B {
    private A a;
    public void setA(A a) {
        this.a = a;
    }
}
public class ReferenceCountingDemo {
    public static void main(String[] args) {
        A a = new A();
        B b = new B();
        a.setB(b);
        b.setA(a);
        // 解除对A和B对象的引用
        a = null;
        b = null;
        // 这里无法回收A和B对象,因为它们之间存在循环引用
        System.gc();

    }
}

在上述案例中,我们创建了两个类A和B,它们分别有一个成员变量用于相互引用。在main方法中,我们创建了一个A对象和一个B对象,并通过setBsetA方法将它们相互引用起来。但是,由于它们之间存在循环引用,即A对象引用B对象,B对象引用A对象,导致它们的引用计数器都不会变为0,无法被回收。

尽管在最后我们将ab设置为null解除了对它们的引用,但由于循环引用的存在,它们的引用计数仍然不为0,无法执行回收操作。

 控制台输出:

从运行结果可以看到内存回收日志包含“Pause Full (System.gc()) 2M->0M(14M) 3.909ms”,意味着虚拟机并没有因为这两个对象互相引用就放弃回收它们,这也从侧面说明了Java虚拟机并不是通过引用计数算法来判断对象是否存活的。

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

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

相关文章

Eclipse使用插件时提示Plugin Error loading shared libraries

项目场景: 使用Eclipse的过程中,依赖openCONFIGURATOR插件进行新建项目时,弹出如下的错误: Plugin Error loading shared libraries 以及具体的信息为: Can’t find dependent libraries 这里我使用的插件为openCONFIGURATOR插件 问题描述 如上场景,提示缺少动态链接库…

Python - 协程基本使用详解【demo】

一. 前言 协程(Coroutine)是一种轻量级的线程,也被称为用户级线程或绿色线程。它是一种用户态的上下文切换方式,比内核态的线程切换更为轻量级,能够高效的支持大量并发操作。 2. 使用协程的好处 Python 中的协程是通…

外卖福利来了,以后都10元以下了,还有机会赚钱,你信吗

怎么实现的点外卖赚钱 在外卖返现平台抢单,用信用卡支付订单,上传好评返现 外卖返现平台是看订单金额,信用卡满减不计入其中,这样就有机会实现赚钱 外卖返现平台 用了半年多返现5000多了,也是一笔开支了 扫最后面的…

快递打单系统使用教程

旅游旺季游客太多怎么办? 相信不少景区特产店都有这种“甜蜜的困扰”。一方面游客多了,自然销售量见长,另一面人流多了,如何服务好顾客,也是特产店的一大难题。 客户询价,太忙没能第一时间回复&#xff1…

Linux15 消息队列 线程

目录 1、进程间通信IPC: 2、多线程 3、向消息队列中写入数据 4、从消息队列中读取数据 5、多线程: 6、将多线程的数据返回给主…

js实现瀑布流布局

jquery.masonry.min.js:https://download.csdn.net/download/weixin_45791806/88224671 jQeasing.js: https://download.csdn.net/download/weixin_45791806/88224673 jquery.lazyload.js这个js可以自己百度下载 直通车:https://download.csd…

【JAVA】我们该如何规避代码中可能出现的错误?(一)

个人主页:【😊个人主页】 系列专栏:【❤️初识JAVA】 文章目录 前言三种类型的异常异常处理JAVA内置异常类Exception 类的层次 前言 异常是程序中的一些错误,但并不是所有的错误都是异常,并且错误有时候是可以避免的&…

设计模式之中介者模式(Mediator)的C++实现

1、中介者模式的提出 在软件组件开发过程中,如果存在多个对象,且这些对象之间存在的相互交互的情况不是一一对应的情况,这种功能组件间的对象引用关系比较复杂,耦合度较高。如果有一些新的需求变化,则不易扩展。中介者…

好用的语音转文字免费app手机软件分享给你

嘿!你有没有遇到过这样的情况:听到了一段精彩的演讲、访谈,但又不方便记录下来?或者,你是一个繁忙的职场人士,需要快速将会议内容转化为文字记录,又苦于手动转写花费时间太多?别担心&#xff0c…

使用Alien对.deb包与.rpm包相互转换

目录 1、切换到root 2、更新yum(更新比较耗时,不更新没试行不,自行斟酌是否跳过这一步) 3、卸载ibus 4、安装Alien及其依赖包 5、安装Alien 6、将.deb转换成.rpm包 7、安装RPM包 8、如果报错 9、将.rpm转换成.deb包 10、安…

白嫖怪小案例———用爬虫实现csdn免费下载资源搜寻

前言 众所周知,在csdn下载资源有很多都是要收费的,最常见的是要积分的 但是小编囊中羞涩,买不起VIP,也没有积分,而资源又要一个一个点进去才知道是不是免费的(最爱0积分了,老白嫖怪了&#xff…

Egg.js构建一个stream流式接口服务

经常需要用到 stream 流式接口服务,比如:大文件下载、日志实时输出等等。本文将介绍如何使用Egg.js构建一个 stream 流式接口服务。 一、准备工作 目录结构: app//controllerindex.jstest.txttest.shindex.js 控制器test.txt 测试文件,最好…

vue3 injection报错 injection“xxx“ not found.

在封装CheckboxGroup组件的的时候&#xff0c;需要通过provide&#xff0c;代码如下&#xff1a; //父组件 <template><div class"envCheckBoxGroup"><slot></slot></div> </template> <script setup> import { provide …

【云原生】3分钟快速在Kubernetes部署Prometheus2.42+Grafana9.5.1+Alertmanager0.25

文章目录 1、简介2、GitHub地址3、环境信息4、安装5、访问Grafana1、简介 Prometheus-operator帮助我们快速创建Prometheus+Grafana+Alertmanager等服务,而kube-prometheus更加完整的帮助我们搭建全套监控体系,这包括部署多个 Prometheus 和 Alertmanager 实例, 指标导出器…

Spring系列篇--关于Spring Bean完整的生命周期【附有流程图,超级易懂】

&#x1f973;&#x1f973;Welcome Huihuis Code World ! !&#x1f973;&#x1f973; 接下来看看由辉辉所写的关于Spring的相关操作吧 目录 &#x1f973;&#x1f973;Welcome Huihuis Code World ! !&#x1f973;&#x1f973; 一.Spring Bean是单例模式还是多例模式 二…

【脚踢数据结构】查找

(꒪ꇴ꒪ )&#xff0c;Hello我是祐言QAQ我的博客主页&#xff1a;C/C语言&#xff0c;Linux基础&#xff0c;ARM开发板&#xff0c;软件配置等领域博主&#x1f30d;快上&#x1f698;&#xff0c;一起学习&#xff0c;让我们成为一个强大的攻城狮&#xff01;送给自己和读者的…

代码随想录—力扣算法题:707设计链表.Java版(示例代码与导图详解)

版本说明 当前版本号[20230818]。 版本修改说明20230818初版 目录 文章目录 版本说明目录707.设计链表思路获取链表第index个节点的数值在链表的最前面插入一个节点在链表的最后面插入一个节点在链表第index个节点前面插入一个节点删除链表的第index个节点 单链表角度总结 7…

STM32入门——IIC通讯

江科大STM32学习记录 I2C通信 I2C&#xff08;Inter IC Bus&#xff09;是由Philips公司开发的一种通用数据总线两根通信线&#xff1a;SCL&#xff08;Serial Clock&#xff09;、SDA&#xff08;Serial Data&#xff09;同步&#xff0c;半双工带数据应答支持总线挂载多设备…

Python“牵手”lazada商品评论数据采集方法,lazadaAPI申请指南

lazada平台API接口是为开发电商类应用程序而设计的一套完整的、跨浏览器、跨平台的接口规范&#xff0c;lazadaAPI接口是指通过编程的方式&#xff0c;让开发者能够通过HTTP协议直接访问lazada平台的数据&#xff0c;包括商品信息、店铺信息、物流信息等&#xff0c;从而实现la…

程序人生:进不了大厂的测试员,究竟还有没有出路了?

金九银十的到来&#xff0c;使得许多职场人按耐不住&#xff0c;纷纷开始找寻合适的工作机会。猎头和HR们摩拳擦掌&#xff0c;争取在这两个月给今年多加点业绩。 对许多互联网人来说&#xff0c;跳槽意味着加薪&#xff0c;而对于程序员而言&#xff0c;是否能跳槽进大厂是他…