对象的内存分配,大部分时间来说,就是在堆空间上的分配(当然也有可能经过JIT编译后被拆散为标量类型并间接地栈上分配),当然,对象分配的规则并不是固定不变的,具体的分配细节取决于 当前使用的是哪一种垃圾回收器的组合、JVM中与内存相关的参数的配置。
1、对象主要是分配在新生代的Eden区上
大多数情况下,对象被分配到新生代的Eden区,当Eden区没有足够的空间区进行分配时,JVM会触发一次Minor GC。
2、启动了本地线程分配缓冲
这种情况下,对象将优先被分配到TLAB(Thread Local Allocation Buffer)中。
3、大对象也可能会直接分配在老年代
所谓的大对象,就是指需要大量连续空间的对象。比如:长字符串、数组。
经常出现大对象,就会导致内存还有空间(碎片化的),但是JVM还是触发了垃圾回收来获取足够大的连续空间来储存它。
JVM提供了一个参数-XX:PretenureSizeThreshold
,用来表示对象的大小达到多少字节就直接储存到老年代,默认值为0字节,表示不管多大都不会直接储存到老年代。
4、长期存活的对象将进入老年代
JVM为每个对象定义了一个年龄计数器。对象在经历过一次Minor GC后,并且能被Survivor区所容纳,那么该对象的年龄+1,当该计数器的值加到了一定的阀值(默认是15岁),该对象就会被转移保存到老年代。该阀值可以通过参数
-XX:MaxTenuringThreshold
来设置。
5、动态对象年龄判断
JVM并不是必需要求对象的年龄达到了
MaxTenuringThreshold
的值才可以被转移到老年代。
如果在Survivor区中,相同年龄的对象大小的总和超过了Survivor区大小的一半,年龄大于或等于该年龄的对象就会被转移到老年代去存储,而无需达到MaxTenuringThreshold
所要求的年龄。
6、空间分配担保
在新生代要发生Minor GC之前,JVM会检查老年代中最大的连续空间是否大于新生代所有对象大小的总和。
1.1、如果结果成立,那么该次Minor GC是安全的。
1.2、如果结果不成立,JVM会参考HandlePromotionFailure
的值来决定是否允许担保失败。
2.1、如果允许冒险,进而会检查老年代中最大的连续空间是否大于前几次晋升到老年代对象总大小的平均值,如果大于平均值,会尝试进行一次Minor GC,该次GC是有一定风险的。
2.2、如果小于平均值或HandlePromotionFailure
设置不允许冒险,那么会执行Full GC。