【从零开始学习JVM | 第六篇】快速了解 直接内存

前言:

当谈及Java虚拟机(JVM)的内存管理时,我们通常会想到堆内存和栈内存。然而,还有一种被称为"直接内存"的特殊内存区域,它在Java应用程序中起着重要的作用。直接内存提供了一种与Java堆内存和本地内存进行交互的方式,同时也为我们提供了更高效的内存管理手段。

在本文中,我们将深入探讨JVM直接内存的概念、工作原理和使用场景。我们将介绍直接内存与Java堆内存的对比,探讨其中的异同点。此外,我们还将探讨如何使用Java NIO库来操作直接内存,以及如何通过合理的使用直接内存来提升应用程序的性能。

目录

前言:

直接内存:

直接内存的优点:

 如何操作直接内存:

JAVA代码方式:

 手动设置方式:

JVM使用场景:

总结:


直接内存:

        在Java虚拟机(JVM)中,直接内存(Direct Memory) 是一块与Java堆独立管理的内存区域。它是通过Java NIO(New I/O)库引入的一种特性。与Java堆相比,直接内存的分配和释放不受Java堆大小限制,因此可以在一些特定场景下提供更高的性能。        

直接内存 在《Java虚拟机规范》中并不存在,所以不属于Java运行时的内存区域。

直接内存提升了IO操作的效率,在传统的IO操作中,我们的执行流程是这样的:

 本地文件先要读取到内存当中,然后再复制到Java堆中,才可以进行读操作。

而如果我们使用直接内存,就不需要再去将数据复制到Java堆中才可以进行读操作,而是在堆中创建一个Java对象的引用。

通过这种方式减少了数据复制的开销,提高了效率。

直接内存的优点:

  1. 无需垃圾回收:直接内存并不受Java堆的垃圾回收机制管理,不会占用宝贵的堆空间,也不会对垃圾回收器产生额外的压力。
  2. 与本地操作系统直接交互:直接内存是通过使用本地方法库(Native Libraries)来与操作系统进行交互的。这意味着可以利用操作系统提供的高效的内存管理功能,如零拷贝等。
  3. 基于操作系统的零拷贝:在使用直接内存进行I/O操作时,可以通过零拷贝技术将数据直接从直接内存传输到网络或磁盘上,避免了数据复制的开销,提高了性能。
  4. 分配和释放成本较高:由于直接内存需要与操作系统进行交互,所以它的分配和释放成本相对较高,而且需要谨慎管理以避免资源泄漏。

 如何操作直接内存:

JAVA代码方式:

        在Java中,可以使用  java.nio.ByteBuffer  类来创建直接内存数据。直接内存是通过操作系统分配的内存,不受 Java 堆内存大小的限制,通常用于需要频繁 I/O 操作的场景。

import java.nio.ByteBuffer;

public class DirectMemoryExample {
    public static void main(String[] args) {
        // 分配直接内存,创建一个容量为 10 字节的 ByteBuffer
        ByteBuffer directBuffer = ByteBuffer.allocateDirect(10);

        // 在直接内存中写入数据
        for (int i = 0; i < 10; i++) {
            directBuffer.put((byte) i);
        }

        // 读取直接内存中的数据
        directBuffer.flip();
        while (directBuffer.hasRemaining()) {
            System.out.print(directBuffer.get() + " ");
        }

        // 释放直接内存,显式调用释放方法
        clean(directBuffer);
    }

    // 释放直接内存的方法
    public static void clean(final ByteBuffer buffer) {
        if (buffer.isDirect()) {
            try {
                sun.misc.Cleaner cleaner = ((sun.nio.ch.DirectBuffer) buffer).cleaner();
                if (cleaner != null) {
                    cleaner.clean();
                }
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    }
}

 手动设置方式:

可以通过 -XX:MaxDirectMemorySize  JVM 参数来手动设置直接内存的最大值。该参数用于限制直接内存的大小,当超出该值时,会抛出OutOfMemoryError异常。

下面是一个示例的 JVM 启动参数:

java -XX:MaxDirectMemorySize=1g MyApplication

这个命令会将直接内存的最大值设置为 1GB,并启动 MyApplication  应用程序。

需要注意的是   -XX:MaxDirectMemorySize   参数只是用于限制直接内存的大小,并不代表实际分配的直接内存大小。例如,如果分配的直接内存大小超过了该参数所设置的上限,也不会立即抛出异常,而是视情况在发生 GC 或者其他操作时释放部分直接内存。因此,在使用直接内存时,需要根据实际情况合理设置该参数的值,避免过度占用系统资源。

JVM使用场景:

  1. 网络编程:在网络通信中,需要频繁地进行数据的读取和发送,如果使用堆内存,就需要将数据先读入堆内存,再复制到网络缓冲区中,这样会增加不必要的开销。使用直接内存,可以直接将数据映射到网络缓冲区,避免数据复制,提高了网络通信的效率。

  2. 文件操作:在读写大文件时,使用传统的文件读写方式,需要将数据先读入堆内存,再进行处理。使用直接内存,可以直接将文件映射到内存中,避免了数据复制,减少了 I/O 操作的次数,提高了文件读写的效率。

  3. 图像处理和多媒体应用:在图像处理和多媒体应用中,需要频繁地进行像素数据的读取和处理。使用直接内存,可以将像素数据映射到内存中,避免了数据复制,提高了图像处理和多媒体应用的效率。

需要注意的是,使用直接内存时需要特别谨慎,因为直接内存是由操作系统分配的,不受 JVM 垃圾回收机制的管理,如果分配过多的直接内存,可能会导致操作系统资源耗尽,从而影响系统的稳定性。因此,在使用直接内存时,需要根据实际情况合理设置直接内存的大小,并在使用完毕后及时释放直接内存。

总结:

        总的来说,JVM 直接内存在现代 Java 应用程序中扮演着重要的角色。它主要用于提高 I/O 操作的效率,特别是在涉及网络通信、文件操作和大规模数据处理时。通过直接内存,可以避免数据的额外复制,直接在操作系统级别操作数据,从而提高了应用程序的性能和吞吐量。

然而,使用直接内存也需要谨慎对待。由于直接内存不受 JVM 垃圾回收机制管理,过度使用可能导致操作系统资源耗尽,影响系统的稳定性。因此,在使用直接内存时,需要合理设置直接内存的大小,并确保及时释放不再需要的直接内存,以避免潜在的问题。

综上所述,合理利用 JVM 直接内存可以有效提升 Java 应用程序的性能,但同时也需要注意资源管理和释放,以确保系统的稳定性和可靠性。

如果我的内容对你有帮助,请点赞,评论,收藏。创作不易,大家的支持就是我坚持下去的动力!

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

十几个软件测试实战项目【外卖/医药/银行/电商/金融】

项目一&#xff1a;ShopNC商城 项目概况&#xff1a; ShopNC商城是一个电子商务B2C电商平台系统&#xff0c;功能强大&#xff0c;安全便捷。适合企业及个人快速构建个性化网上商城。 包含PCIOS客户端Adroid客户端微商城&#xff0c;系统PC后台是基于ThinkPHP MVC构架开发的跨…

windows如何解决端口冲突(实用篇)

在项目设计中&#xff0c;环境配置成功点击运行瞬间&#xff0c;一大堆红爆出&#xff0c;8080端口占用&#xff0c;这个是很烦人的。。。 解决方式&#xff1a; 笨方法&#xff1a;一、查看所有端口实用情况&#xff08;挨个扫&#xff09; 按住【WINR】快捷键打开运行输入…

Android View闪烁动画AlphaAnimation,Kotlin

Android View闪烁动画AlphaAnimation&#xff0c;Kotlin private fun flickerAnimation(view: View?) {val animation: Animation AlphaAnimation(1f, 0f) //不透明到透明。animation.duration 500 // 1次过程时长。animation.interpolator LinearInterpolator() // 线性速…

赴美上市传闻再起,SHEIN走到十字路口

作者 | 辰纹 来源 | 洞见新研社 裹挟着“黑五”大胜的余波&#xff0c;跨境电商巨头SHEIN&#xff08;希音&#xff09;将赴美IPO的传闻又在行业中散播开来。 金融投资报称SHEIN此次IPO的估值或达900亿美元&#xff1b;上海证券报表示&#xff0c;SHEIN已对投资人发出了路演…

10天玩转Python第3天:python循环语句和字符串、列表全面详解与代码示例

目录 1 循环1.1 for 循环1.2 break 和 continue 2 容器3 字符串3.1 定义3.2 下标3.3 切片3.4 字符串的查找方法 find3.5 字符串的替换方法 replace3.6 字符串的拆分 split3.7 字符串的链接 join 4 列表4.1 定义4.1 列表支持下标和切片, 长度4.3 查找 - 查找列表中数据下标的方法…

centos7x 安装支持gpu驱动的docker

1、卸载以前版本的驱动 sudo /usr/bin/nvidia-uninstall2、先安装基础项 yum install kernel kernel-devel gcc make -yyum install kernel kernel-devel gcc gcc-c make -y 3、禁用驱动源 nouveau echo "blacklist nouveau " >>/etc/modp…

12.12_黑马数据结构与算法笔记Java

目录 079 优先级队列 无序数组实现 080 优先级队列 有序数组实现 081 优先级队列 堆实现 1 082 优先级队列 堆实现 2 083 优先级队列 堆实现 3 084 优先级队列 e01 合并多个有序链表1 084 优先级队列 e01 合并多个有序链表2 085 阻塞队列 问题提出 086 阻塞队列 单锁实…

Linux:gdb的简单使用

个人主页 &#xff1a; 个人主页 个人专栏 &#xff1a; 《数据结构》 《C语言》《C》《Linux》 文章目录 前言一、前置理解二、使用总结 前言 gdb是Linux中的调试代码的工具 一、前置理解 我们都知道要调试一份代码&#xff0c;这份代码的发布模式必须是debug。那你知道在li…

【代码随想录】刷题笔记Day34

前言 考过概率论&#xff0c;发过一场烧&#xff0c;兜兜转转又一月&#xff0c;轻舟已撞万重山&#xff0c;赶紧刷题 贪心算法理论基础 贪心的本质&#xff1a;局部最优→全局最优无套路&#xff0c;常识性推导 举反例 455. 分发饼干 - 力扣&#xff08;LeetCode&#xf…

FastAPI之响应模型

前言 响应模型我认为最主要的作用就是在自动化文档的显示时&#xff0c;可以直接给查看文档的小伙伴显示返回的数据格式。对于后端开发的伙伴来说&#xff0c;其编码的实际意义不大&#xff0c;但是为了可以不用再额外的提供文档&#xff0c;我们只需要添加一个 response_mod…

慢SQL诊断

最近经常遇到技术开发跑来问我慢SQL优化相关工作&#xff0c;所以干脆出几篇SQL相关优化技术月报&#xff0c;我这里就以公司mysql一致的5.7版本来说明下。 在企业中慢SQL问题进场会遇到&#xff0c;尤其像我们这种ERP行业。 成熟的公司企业都会有晚上的慢SQL监控和预警机制。…

784. 字母大小写全排列

字母大小写全排列 描述 : 给定一个字符串 s &#xff0c;通过将字符串 s 中的每个字母转变大小写&#xff0c;我们可以获得一个新的字符串。 返回 所有可能得到的字符串集合 。以 任意顺序 返回输出。 回文串 是正着读和反着读都一样的字符串。 题目 : LeetCode 784. 字母…

Linux--操作系统

1. 常见的操作系统 Windowsmac OSLinuxiOSAndroid 2. 操作系统的定义 操作系统直接运行在计算机上的系统软件&#xff0c; 它是控制硬件和支持软件运行的计算机程序。 3. 操作系统的作用 向下控制硬件向上支持软件的运行&#xff0c;具有承上启下的作用。 4.总结 操作系统…

集合03 Collection (List) - Java

List ArrayListArrayList注意事项ArrayList底层操作机制-源码分析&#xff08;重点&#xff09; VectorVector基本介绍 ——Vector和ArrayList比较Vector底层结构和源码分析 LinkedList基本介绍LinkedList的底层结构和操作机制LinkedList的增删改查 ——LinkedList和ArrayList比…

12.字符串拼接【2023.12.4】

1.问题描述 我们在编程过程中经常会遇到把不同字符串拼接在一起的情况&#xff0c;从而更直观地展示给用户我们所要表达的信息。本题将给出两个字符串&#xff0c;请依次将这两个字符串拼接在一起。 2.解决思路 用字符串拼接符 进行连接两个字符串 3.代码实现 str1input(…

我的创作三周年纪念日

今天收到CSDN官方的来信&#xff0c;创作三周纪念日到了。 Dear: Hann Yang &#xff0c;有幸再次遇见你&#xff1a; 还记得 2020 年 12 月 12 日吗&#xff1f; 你撰写了第 1 篇技术博客&#xff1a; 《vba程序用7重循环来计算24》 在这平凡的一天&#xff0c;你赋予了它…

【异常解决】SpringBoot + Maven 在 idea 下启动报错 Unable to start embedded Tomcat(已解决)

Unable to start embedded Tomcat&#xff08;已解决&#xff09; 一、背景介绍二、原因分析2.1 网络上整理2.2 其他原因 三、解决方案 一、背景介绍 spring boot(v2.5.14) maven idea 启动项目 之前项目一直启动的好好的&#xff0c;都能正常运行。重启的时候突然就不能启…

鸿蒙(HarmonyOS)应用开发——简易版轮播图

简述 轮播图在应用中&#xff0c;已经很常见的展现方式。像uniapp、iview&#xff0c;viewUI等前端组件框架&#xff0c;都提供了轮播图组件。那么在harmonyOS中&#xff0c;如果要实现轮播&#xff0c;我们是使用swiper 组件 swiper组件 swiper 组件是一种容器组件。它提供…

Linux---虚拟机软件

1. 虚拟机软件的介绍 它是能够虚拟出来计算机的一个软件。 常用虚拟机软件: VmwareVirtualBox 说明: 只有安装了虚拟机软件才可以创建虚拟机&#xff0c;当然通过虚拟机软件还可以创建多个虚拟机。 2. 虚拟机的介绍 就是模拟一个真实的计算机&#xff0c;好比一个虚拟的…

DRBD分布式存储实验

DRBD DRBD的全称为&#xff1a;Distributed Replicated Block Device (DRBD) 分布式块设备复制 与心跳连接结合使用&#xff0c;构建高可用性(HA)的集群。 实现方式是通过网络来镜像(mirror)整个设备。它允许用户在远程机器上建立一个本地块设备的实时镜像。DRBD负责接收数据…