Java堆
- 对于Java应用程序来说,Java堆 是虚拟机所管理的内存中最大的一块。
- 堆,是被所有线程共享的一块内存区域,在虚拟机启动时创建。
- 堆,唯一的目的就是存放对象实例,Java世界中“几乎”所有的对象实例都是在这里分配内存的。在《Java虚拟机规范》中对Java虚拟机的描述是:“所有的对象实例以及数组都应当在堆上分配”。但随着语言的不断进步,即使只考虑现在,由于技术编译技术的进步,尤其是逃逸分析技术的日渐强大,栈上分配。标量替换优化手段已经导致一些微妙的变化,所以说Java对象实例都分配在堆上变得不那么绝对了。
- 堆是垃圾收集器管理的内存区域,因此有一些资料中也称作“GC堆”(Garbage Collected Heap)。国内尽然没翻译成垃圾堆。
- 从回收内存角度看,由于现代垃圾收集器大部分都是基于分代手机理论设计的,所以Java堆中经常对出现“新生代”、“老年代”、“永久代”,“Eden空间”、“From Survivor空间”、“To Survivor空间”等名词。这些区域的划分仅仅是一部分垃圾收集器的共同特性或者说设计风格而已,并非某个Java虚拟机具体实现的固有内存布局,也不是对Java堆的进一步细致划分。
- 不少资料上经常谢了类似于“Java虚拟机的内存分为新生代、老年代、永生代…”这样的内容,放在十年前或许不会有太大歧义,但是到了今天,垃圾收集器技术已经与十年前不可同日而语,再按照上诉说法的话,就有许多值得商榷的地方了。
- 从分配内存的角度看,所有线程共享的Java堆可以划分为多个线程私有的分配缓冲区(Thread Local Allocation Buffer,TLAB),以提高对象分配效率。
- 不管从什么角度,无论如何划分,都不会改变Java堆存储内容的共性。将Java堆细分的目的是为了更好的回收内存,或者更快的分配内存。
- Java堆可以被实现成固定大小的,也可以是可扩展的。当前流行的Java虚拟机都是按照可扩展来实现的(通过参数-Xmx和-Xms设定)