直接内存 直接内存的释放和回收

article/2025/11/1 11:50:55

直接内存

特点

  • 不属于Java虚拟机管理,属于系统内存;
  • 属于操作系统,常见于NIO操作时,比如ByteBuffer【】用于数据缓冲区
  • 分配回收成本较高,但读写性能高;
  • 不受JVM内存回收管理

文件读写过程

java不具备磁盘读写的能力,因此需要调用操作系统提供的方法;java函数->本地方法用户态->内核态

弊端:需要在系统内存和java堆内存都复制一份磁盘文件,浪费空间也降低传输效率

在这里插入图片描述

直接内存是在系统内存和Java堆内存之间开辟出的一块共享区域,可以供操作系统和java代码访问。

好处:只需要缓冲一份磁盘文件,效率更高

在这里插入图片描述

内存溢出

​ 直接内存也会存在内存溢出的问题,比如下面的例子,在循环中不断申请一个100m的直接内存空间,把其加入list中,最终会造成直接内存不够,抛出直接内存溢出异常

在这里插入图片描述

可以在任务管理中查看到当前直接内存的大小

在这里插入图片描述

直接内存释放原理

java底层使用Unsafe类来分配释放直接内存,一般是jdk内部使用

在这里插入图片描述

//通过ByteBuffer申请100M的直接内存
//容量默认单位为字节Byte
int _100M = 1024*1024;   
ByteBuffer byteBuffer=ByteBuffer.allocateDirect(_100M);

allocateDirect实现过程

/*** Allocates a new direct byte buffer.* @param  capacity*         The new buffer's capacity, in bytes*/
public static ByteBuffer allocateDirect(int capacity) {   return new DirectByteBuffer(capacity);
}

DirectByteBuffer实现过程

  • 调用 UNSAFE.setMemory(base, size, (byte) 0);来分配直接内存
  • cleaner = Cleaner.create(this, new Deallocator(base, size, cap));
  • 调用 UNSAFE.freeMemory(address);来释放直接内存
// Primary constructor
//
DirectByteBuffer(int cap) {                   // package-privatesuper(-1, 0, cap, cap, null);boolean pa = VM.isDirectMemoryPageAligned();int ps = Bits.pageSize();long size = Math.max(1L, (long)cap + (pa ? ps : 0));Bits.reserveMemory(size, cap);long base = 0;try {base = UNSAFE.allocateMemory(size);} catch (OutOfMemoryError x) {Bits.unreserveMemory(size, cap);throw x;}UNSAFE.setMemory(base, size, (byte) 0);if (pa && (base % ps != 0)) {// Round up to page boundaryaddress = base + ps - (base & (ps - 1));} else {address = base;}cleaner = Cleaner.create(this, new Deallocator(base, size, cap));att = null;}

Deallocator的实现过程

public void run() {if (address == 0) {// Paranoiareturn;}UNSAFE.freeMemory(address);address = 0;Bits.unreserveMemory(size, capacity);
}

cleaner是虚引用类型,当被关联的对象(ByteBuffer)被java垃圾回收时,会触发cleaner的clean方法,执行 Deallocator任务对象的run方法,即最终调用 UNSAFE.freeMemory(address);

在这里插入图片描述

禁用显示回收堆直接内存的影响

代码中的System.gc是****显示垃圾回收,是一次Full GC 比较耗时,影响程序执行;可以使用命令**禁用显示回收,**但是会造成ByteBuffer对象存活,直接内存无法释放,造成长时间得不到释放的问题。

解决方法:直接使用Unsafe调用FreeMemory手动释放 UNSAFE.freeMemory(address);

可以使用命令**禁用显示回收,**但是会造成ByteBuffer对象存活,直接内存无法释放,造成长时间得不到释放的问题。

解决方法:直接使用Unsafe调用FreeMemory手动释放 UNSAFE.freeMemory(address);

在这里插入图片描述


http://chatgpt.dhexx.cn/article/99j6VHhm.shtml

相关文章

解决The‘Access-Control-Allow-Origin‘ header contains multiple values‘*, ....‘, but only one is allowed

报错内容: Access to XMLHttpRequest at ‘http://www.z…n.com/api/login’ from origin ‘http://z…n.com’ has been blocked by CORS policy: The ‘Access-Control-Allow-Origin’ header contains multiple values ‘*, http://z…n.com’, but only one is …

C++中std::allocator的使用

标准库中包含一个名为allocator的类,允许我们将分配和初始化分离。使用allocator通常会提供更好的性能和更灵活的内存管理能力。 new有一些灵活性上的局限,其中一方面表现在它将内存分配和对象构造组合在了一起。类似的,delete将对象析构和内…

alloc的流程

1⃣️ 2⃣️ 3⃣️ 现在我们看的objc源码都是2.0的版本,之前还有一个1.0的版本 4⃣️ 5⃣️核心方法 最主要的就是最下面三个方法 他的最重要作用就是开辟内存 cls->instanceSize 先计算出需要的内存空间大小这个大小只和对象的成员变量有关 在没有成员变量…

ByteBuffer.allocate()与allocateDirect()的区别

allocate()产生的是HeapByteBuffer的实例, 本质上是一个no direct buffer, allocateDirect()产生的是DirectByteBuffer的实例, 本质是一个direct buffer 主要区别 buffer的创建方式不同, no direct buffer还可以通过封装已存在的byte array来产生执行IO操作时不同, no direct …

ByteBuffer常用方法与分析

目录 目标 常用API 工具方法 演示案例 allocate(int capacity)和allocateDirect(int capacity) put()和get() flip()和hasRemaining() clear() compact() wrap() 总结 目标 掌握ByteBuffer常用方法,分析ByteBuffer对象在切换读写模式的情况下基本属性的变…

allocate与allocateDirect的性能测试

allocate与allocateDirect的性能测试 测试工具JMH测试代码JMH结果结论 测试工具JMH java基准测试框架 测试代码 直接分配系统内存(allocateDirect) -测试申请内存性能JVM堆分配内存(allocate)-测试申请内存性能直接内存-操作-连续二十次添加(allocateDirect)-测试内存操作性能…

ByteBuffer.allocate()与ByteBuffer.allocateDirect()方法的区别

在Java中当我们要对数据进行更底层的操作时,一般是操作数据的字节(byte)形式,这时经常会用到ByteBuffer这样一个类。ByteBuffer提供了两种静态实例方式: public static ByteBuffer allocate(int capacity) public s…

8、ByteBuffer(方法演示2(allocate堆内存和allocateDirect直接内存))

ByteBuffer(方法演示2(allocate堆内存和allocateDirect直接内存)) Allocate:java堆内存:读写效率低,收到gc的影响(因为我们的java对象也是存在堆内存的) !!…

03 Java NIO allocateDirect()和allocate()区别

03 Java NIO allocateDirect和allocate区别 allocateDirect()和allocate()区别直接与非直接缓冲区直接与非直接缓冲区 源码分析 allocateDirect()和allocate()区别 allocateDirect:分配直接缓冲区,将缓冲区简历在物理内存中,可以提交效率 all…

ByteBuffer.allocate()与ByteBuffer.allocateDirect()方法的区别。

在Java中当我们要对数据进行更底层的操作时,一般是操作数据的字节(byte)形式,这时经常会用到ByteBuffer这样一个类。ByteBuffer提供了两种静态实例方式: public static ByteBuffer allocate(int capacity) public …

ByteBuffer中的allocate和allocateDirect

1、区别 allocate方法创建的内存在jvm的管理范围,而allocateDirect方法创建的内存不由jvm管理,意思就是allocateDirect创建的内存由系统直接管理 2、释放 allocateDirect创建的内存不归jvm管理,那如何释放呢?虽然这块内存不属于…

NIO ByteBuffer的allocate与allocateDirect区别(HeapByteBuffer与DirectByteBuffer的区别)

参考:https://blog.csdn.net/zhxdick/article/details/81084672 其中allocateDirect分配的字节缓冲区用中文叫做直接缓冲区(DirectByteBuffer),用allocate分配的ByteBuffer叫做堆字节缓冲区(HeapByteBuffer).. 其实根据类名就可…

【系统分析师】论文

文章目录 一、论文框架二、相关素材概览2.1 常见开发模型2.2 SOA架构 三、注意事项3.1 岗位职责3.2 项目背景3.3 理论部分讨论3.4 主体内容3.5 结论部分 四、范文4.1 开发模型主题反例正例 4.2 需求获取技术反例正例 4.3 系统测试反例正例 4.4反例正例 一、论文框架 二、相关素…

系统分析员论文12篇合集

系分论文1 企业人事信息系统的应用 【摘要】 本文讨论《企业人事信息系统》项目的需求分析方法与工具的选用。该系统的建设目标是帮助该企业管理好企业内部的人员和人员的活动,人事信息管理指的是企业员工从招聘面试到离职退休的全过程,涉及的主要活动…

系统分析师近几年论文走向

下面是整理的近几年的系统分析师考试论文题目

Java小游戏-俄罗斯方块

摘 要 随着时代的不断发展,个人电脑也在不断普及,一些有趣的桌面游戏已经成为人们在使用计算机进行工作或工作之余休闲娱乐的首选,从最开始的Windows系统自带的黑白棋、纸牌、扫雷等游戏开始,到现在目不暇接的各种游戏&#xff0…

Java版本实现俄罗斯方块小游戏

效果预览图 视频预览: 俄罗斯方块 代码部分 import javax.swing.*; import java.awt.*; import java.awt.event.*; import java.util.Random;public class Block0011 extends JFrame implements KeyListener, ActionListener, MouseListener {// 记分int fenshu…

基于JAVA的俄罗斯方块游戏的设计与实现

目 录 1 引言 1.1 课题研究背景和意义 1.2 国内外现状分析 1.3 课题研究主要内容 1.4 论文组织结构 2 核心技术介绍 2.1 AWT技术介绍 2.2 eclipse编译器介绍 2.3 JAVA集合类技术简介 3 需求分析 3.1 游戏的设计模式 3.2 系统的设计目标 3.3 经济可行…

Java版俄罗斯方块

Java版俄罗斯方块 08年写的一个Java版俄罗斯方块程序 界面做的中规中矩,每种形状颜色都不相同 程序控制还可以,没什么大的Bug 消磨时间的时候可以Down下来玩玩 下载链接:http://download.csdn.net/user/zhaohuihua 菜单选项 也做了不少菜…