JVM的堆 (Heap) 和非堆 (Non-heap) 内存

堆 (Heap) 和非堆 (Non-heap) 内存

按照官方的说法:

“Java 虚拟机具有一个堆,堆是运行时数据区域,所有类实例和数组的内存均从此处分配。堆是在 Java 虚拟机启动时创建的。 ”
“ 在 JVM 中堆之外的内存称为非堆内存 (Non-heap memory)” 。

可以看出 JVM 主要管理两种类型的内存:堆和非堆。简单来说堆就是 Java 代码可及的内存,是留给开发人员使用的;非堆就是 JVM留给自己用的,所以方法区、 JVM 内部处理或优化所需的内存 ( 如 JIT 编译后的代码缓存 ) 、每个类结构 ( 如运行时常数池、字段和方法数据 )以及方法和构造方法的代码都在非堆内存中。

堆内存分配
JVM 初始分配的内存由 -Xms 指定,默认是物理内存的 1/64 ; JVM最大分配的内存由 -Xmx 指定,默认是物理内存的 1/4 。默认空余堆内存小于 40% 时, JVM 就会增大堆直到 -Xmx 的最大限制;空余堆内存大于70% 时, JVM 会减少堆直到 -Xms 的最小限制。 因此服务器一般设置 -Xms 、 -Xmx 相等以避免在每次 GC 后调整堆的大小。

非堆内存分配

JVM 使用 -XX:PermSize 设置非堆内存初始值,默认是物理内存的 1/64 ;由 -XX:MaxPermSize设置最大非堆内存的大小,默认是物理内存的 1/4 。

JVM 内存限制 ( 最大值 )

首先 JVM 内存限制于实际的最大物理内存,假设物理内存无限大的话, JVM 内存的最大值跟操作系统有很大的关系。简单的说就 32 位处理器虽然可控内存空间有 4GB,但是具体的操作系统会给一个限制,这个限制一般是 2GB-3GB (一般来说 Windows 系统下为 1.5G -2G , Linux 系统下为 2G-3G ),而 64bit 以上的处理器就不会有限制了。