一.什么是堆
Heap堆是JVM所管理的内存中最大的一块区域,被所有线程共享的一块内存区域,堆中存放着对象实例,几乎所有的对象实例以及数组都在这里分配内存,从JDK1.7开始默认开启逃逸分析,如果某些方法中的对象引用没有被返回或者未被外面使用(也就是未逃逸出去),那么对象可以直接在栈上分配内存。
二.堆区分代划分(新生代和老年代)
堆区划分为新生代和老年代,新生代占堆区的1/3,而新生代中又分为Eden区(占新生代区的8/10),和Survivor区(由from和to两块内存区域组成,分别占新生代区的1/10),老年代则占堆区的2/3。
三.创建对象是如何在堆中分配内存
首先,当我们创建一个对象时,对象会在新生代区的Eden区生成,如果Eden区内存未满,则直接分配内存空间,如果Eden区内存已满,则会触发YGC垃圾回收,将Eden区中没有被引用的对象直接回收,没有被回收的对象会被送至Survivor区,每次YGC垃圾回收的时候,都会将对象放置当前Survivor区中的某一块未使用的内存区域,然后再将当前正在使用的内存区域(也就是Survivor区中另一块内存区域)清除,交换两块内存空间的状态,如果YGC移送的对象大于Survior区的容量上限,则会直接放置老年区,如果老年区放得下,则分配内存空间,如果放不下,触发FGC垃圾回收,回收完成后,再次判断老年代是否放得下,如果还是放不下,则会触发OOM错误。
注:一个对象从新生代晋升到老年代的阈值默认是15,也就是说当某个对象在新生代的Survivor区的两块内存区域中交换14次之后,就会晋升老年代。