【JVM】10、直接内存

参考

native memory 和direct memory

这部分参考

  • JVM管理的内存可以总体划分为两部分:Heap Memory和Native Memory。
  • Native Memory和Off-heap是一个概念,没有相应的参数来控制大小,其大小依赖于操作系统进程的最大值。
  • direct memory 意味着用java.nio.DirectByteBuffer.的方式用native memory
  • Native heap means you use native memory by means of unsafe.allocateMemory or simply do a malloc in your JNI code.

metaspace 是存在native memory的,是有GC的。而direct memory是没有gc的

native memory 存什么

1.管理java heap的状态数据(用于GC);
2.JNI调用,也就是Native Stack;
3.JIT(即使编译器)编译时使用Native Memory,并且JIT的输入(Java字节码)和输出(可执行代码)也都是保存在Native Memory;
4.NIO direct buffer。对于IBM JVM和Hotspot,都可以通过-XX:MaxDirectMemorySize来设置nio直接缓冲区的最大值。默认是64M。超过这个时,会按照32M自动增大。
5.对于IBM的JVM某些版本实现,类加载器和类信息都是保存在Native Memory中的。



直接内存

  1. 直接内存不是虚拟机运行时数据区的一部分,也不是《Java虚拟机规范》中定义的内存区域。
  2. 直接内存是在Java堆外的、直接向系统申请的内存区间。
  3. 来源于NIO,通过存在堆中的DirectByteBuffer操作Native内存
  4. 通常,访问直接内存的速度会优于Java堆。即读写性能高。
  5. 因此出于性能考虑,读写频繁的场合可能会考虑使用直接内存。
  6. Java的NIO库允许Java程序使用直接内存,用于数据缓冲区

BIO 与NIO

NIO

IO NIO
是最初的讲如何实现数据传输的规范 NIO(New IO/Non-Blocking IO)
阻塞式的 非阻塞式的
基础传输工具 byte[]/char[] Buffer
Stream Channel

NIO是通过DirectByBuffer操作Native内存的

在这里插入图片描述
allocateDirect方法帮我们创建了直接内存(DirectByBuffer就是直接本地内存的对象
在这里插入图片描述

非直接缓存区(BIO)

原来采用BIO的架构,在读写本地文件时,我们需要从用户态切换成内核态
在这里插入图片描述

直接缓冲区(NIO)

NIO 直接操作物理磁盘,省去了中间商赚差价
在这里插入图片描述

直接内存与 OOM

直接内存大小设置

直接内存大小可以通过MaxDirectMemorySize设置
如果不指定,默认与堆的最大值-Xmx参数值一致

java.lang.OutOfMemoryError: Direct buffer memory

直接内存也可能导致OutofMemoryError异常

在这里插入图片描述
直接内存和Java堆没有关系,但是系统内存是有限的,Java堆和直接内存的总和依然受限于操作系统能给出的最大内存。
在这里插入图片描述

直接内存缺点

分配回收成本较高
不受JVM内存回收管理

JVM内存结构

简单理解java process memory = java heap + native memory
可以把栈、PC寄存器那些的忽略不计
在这里插入图片描述

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章