堆(Heap)
Heap
堆区,用于存放对象实例和数组
Heap
堆是JVM
所管理的内存中最大的一块区域,被所有线程共享的一块内存区域。堆区中存放对象实例,“几乎”所有的对象实例以及数组都在这里分配内存。
Java 世界中“几乎”所有的对象都在堆中分配,但是,随着
JIT
编译器的发展与逃逸分析技术逐渐成熟,栈上分配、标量替换优化技术将会导致一些微妙的变化,所有的对象都分配到堆上也渐渐变得不那么“绝对”了。从 JDK 1.7 开始已经默认开启逃逸分析,如果某些方法中的对象引用没有被返回或者未被外面使用(也就是未逃逸出去),那么对象可以直接在栈上分配内存。
新生代、老年代重要
Heap
堆是垃圾收集器GC
(Garbage Collected
)管理的主要区域,因此堆区也被称作GC
堆(Garbage Collected Heap
)。从垃圾回收的角度,由于现在收集器基本都采用分代垃圾收集算法,所以 JVM
中的堆区往往进行分代划分,例如:新生代 和 老年代。目的是更好地回收内存,或者更快地分配内存。
创建对象的内存分配
创建一个新对象,在堆中的分配内存。
大部分情况下,对象会在 Eden
区生成,当 Eden
区装填满的时候,会触发 Young Garbage Collection
,即 YGC
垃圾回收的时候,在 Eden
区实现清除策略,没有被引用的对象则直接回收。
依然存活的对象会被移送到 Survivor
区。Survivor
区分为 s0
和 s1
两块内存区域。每次 YGC
的时候,它们将存活的对象复制到未使用的Survivor
空间(s0
或 s1
),然后将当前正在使用的空间完全清除,交换两块空间的使用状态。每次交换时,对象的年龄会加+1
。
如果 YGC
要移送的对象大于 Survivor
区容量的上限,则直接移交给老年代。一个对象也不可能永远呆在新生代,在 JVM
中 一个对象从新生代晋升到老年代的阈值默认值是 15
,可以在 Survivor
区交换 14 次之后,晋升至老年代。
堆区最容易出现的就是 OutOfMemoryError
错误,这种错误的表现形式会有以下两种:
OutOfMemoryError: GC Overhead Limit Exceeded
: 当JVM
花太多时间执行垃圾回收,并且只能回收很少的堆空间时,就会发生此错误。OutOfMemoryError: Java heap space
:假如在创建新的对象时, 堆内存中的空间不足以存放新创建的对象, 就会引发此错误。
此种情况,与配置的最大堆内存有关,且受制于物理内存大小。