对象优先在Eden
分配
大多数情况下,对象在新生代Eden
区中分配。但Eden
区没有足够的空间进行分配时,虚拟机将发生一次Minor GC
。如果当GC发现对象无法全部放入Survivor
空间,就会通过分配担保机制提前转移到老年代中。
大对象直接进入老年代
大对象指,需要大量连续内存空间的Java对象,最典型的就是那种很长的字符串以及数组。
虚拟机提供一个-XX:PretenureSizeThreshold
参数,令大于这个设置值的对象直接在老年代分配。这样做目的是避免在Eden
区以及两个Survivor
区之间发生大量的内存复制。
长期存活的对象将进入老年代
虚拟机给每个对象定义一个对象年龄(Age
)计数器。如果对象在Eden
出生并经过第一次Minor GC
后仍然存活,并且能被Survivor
容纳的话,将被移动到Survivor
空间中,并且对象年龄设为1.对象在Survivor
区中每熬过一次Minor GC
,年龄就增长1岁,当年龄增加到一定程度(默认15岁),就会被晋升到老年代中。可以通过参数-XX:MaxTenuringThreshold
设置年龄阈值。
动态对象年龄判断
虚拟机并不是永远要求对象的年龄必须达到MaxTenuringThreshold
才能晋升老年代,如果在Survivor
空间相同年龄所有对象大小的综合大于Survivor
空间的一半,年龄大于等于该年龄的对象就可以直接进入老年代。
空间分配担保
在发生Minor GC
之前,虚拟机会先检查老年代最大可用的连续空间是否大于新生代所有对象总空间,如果成立,那么Minor GC
确保是安全的。如果不成立,则虚拟机会查看HandlePromotionFailure
设置值是否允许担保失败。如果允许,那么会继续检查老年代最大可用的连续空间是否大于历次晋升到老年代对象的平均大小,如果大于,将尝试着进行一次Minor GC
,尽管这次Minor GC
是有风险的;如果小于,或者HandlePromotionFailure
设置为不允许冒险,这时改为进行一次Full GC
。
该文章来源《深入理解Java虚拟机》