JVM之内存分配策略

 对象的内存分配,往大方向讲,就是在堆上分配,对象主要分配在新生代的Eden区上,如果启动了本地线程分配缓存,将按照线程优先在TLAB上分配。少数情况下也可能会直接分配在老年代中,如大对象(需要大量连续内存空间的Java对象)直接进入老年代。
 在这之前先来简单了解一下JVM的内存回收策略。当前虚拟机都采用了“分代收集”算法,此算法根据对象存活周期的不同将内存划分为几块。一般是把Java堆分为新生代和老年代,这样就可以根据各个年代的特点采用最适当的算法。

 在讲内存分配之前,先来了解一下Minor GC、Major GC与Full GC的区别。

  • Minor GC:指发生在新生代的垃圾收集动作,因为Java对象大多都具备朝生夕灭的特征,所以Minor GC非常频繁,一般回收速度也比较快,当Eden区满时,触发该操作
  • Major GC:Major GC指发生在老年代的GC,出现了Major GC,经常会伴随至少一次的Minor GC,Major GC的速度一般会比Minor GC慢10倍以上。
  • Full GC:Full GC是对于整个堆来进行的GC,触发条件是:System.gc()、老年代空间不足、方法区空间不足(针对1.7及之前版本)、统计得到的Minor GC晋升到老年代的平均大小大于老年代的剩余空间、堆中分配很大的对象
对象优先在Eden分配

 大多数情况下,对象在新生代Eden区中分配。当Eden区没有足够空间进行分配时,虚拟机将发起一次Minor GC。

大对象直接进入老年代

 所谓大对象是指,需要大量连续内存空间的Java对象,最典型的大对象是那种很长的字符串及数组。大对象对虚拟机的内存分配来说就是一个坏消息,经常出现大对象容易导致内存还有不少空间时就提前触发垃圾收集以获取足够的连续空间来“安置”它们。

长期存活对象将进入老年代

 虚拟机给每个对象定义了一个对象年龄计数器,如果对象在Eden出生并经过一个Minor GC后仍然存活,并且能被Survivor容纳的话,将被移动到Survivor空间,并且对象的年龄设为1。对象在Survivor区中每“熬过”一次Minor GC,年龄将增加1岁,当它的年龄增加到一定程度(默认为15岁),就将会被晋升到老年代。对象晋升老年代的年龄阈值,可以通过参数-XX:MaxTenuringThreshold设置。

动态对象年龄判断

 为了能更好的适应不同程序的内存状况,虚拟机并不是永远地要求对象的年龄必须达到MaxTenuringThreshold才能晋升,如果Survivor空间中相同年龄所有对象大小的总和大于Survivor空间的一半,年龄大于或等于该年龄的对象就可以直接进入老年代,无须等到MaxTenuringThreshold中要求的年龄。

空间分配担保

 在发生Minor GC之前,虚拟机会先检查老年代最大可用的连续空间是否大于新生代所有对象总空间,如果这个条件成立,那么Minor GC可以确保是安全的。如果不成立,则虚拟机会查看HandlePromotionFailure设置值是否允许担保失败。如果允许,那么会继续检查老年代最大可用的连续空间是否大于历次晋升到老年代对象的平均大小,如果大于,将尝试进行一次Minor GC,尽管这次Minor GC是有风险的;如果小于,或者HandlePromotionFailure设置不允许冒险,那这时也要改为进行一次Full GC。
 新生代使用了复制收集算法,但为了内存使用率,只使用其中一个Survivor空间作为轮换备份,因此当出现大量对象在Minor GC后仍然存活的情况,就需要老年代进行分配担保,把Survivor无法容纳的对象直接进入老年代。老年代要进行这样的担保,前提是老年代本身还有容纳这些对象的剩余空间,一共有多少对象会活下来在内存回收之前是无法明确知道的,所以只好取之前每一次回收晋升到老年代对象容量的平均值作为经验值,与老年代的剩余空间进行比较,决定是否进行Full GC来让老年代腾出更多空间。
 在Java后续版本中,HandlePromotionFailure将不会被使用了,也就是说规则变为只要老年代的连续空间大于新生代对象总大小或者历次晋升的平均值大小就会进行Minor GC,否则将进行full GC。

【参考资料】
《深入理解Java虚拟机》
从实际案例聊聊Java应用的GC优化

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值