一个工作了7年的粉丝去京东面试,遇到一个很有意思的问题。
这是一个关于JVM底层相关的问题,如果平时没有去花时间是肯定回答不出来的。
今天给大家分享的这道面试题是:“JVM分代年龄为什么是15次,可以是25次吗?”
另外我花了1个多星期的时间,把往期高手回答整理成了10W字的文档,想获取的小伙伴可以扫描文章底部二维码领取
下面看看高手对这个问题的回答。
大家记得点赞收藏加关注
需要高手面试文档面试文档的小伙伴可以扫描文章底部二维码
高手:
首先,在JVM的heap内存里面,分为Eden Space、Survivor Space、Old Generation。
当我们在Java里面使用new关键字创建一个对象的时候,JVM会在Eden Space分配一块内存空间来存储这个对象。
当Eden Space的内存空间不足的时候,会触发Young GC进行对象回收。
那些因为存在引用关系而无法回收的对象,JVM会把它们转移到Survivor Space。
Survivor Space内部又分为From区和To区,刚从Eden区转移过来的对象会分配到From区,每经历一次Young GC,这些没有办法被回收的对象就会在From区和To区来回移动,每移动一次,这个对象的GC年龄就加1。默认情况下GC年龄达到15的时候,JVM就会把这个对象移动到Old Generation。
其次呢,一个对象的GC年龄,是存储在对象头里面的,一个Java对象在JVM内存中的布局由三个部分组成,分别是对象头、实例数据、对齐填充。而对象头里面有4个bit位来存储GC年龄。
而4个bit位能够存储的最大数值是15,所以从这个角度来说,JVM分代年龄之所以设置成15次是因为它最大能够存储的数值就是15。
虽然JVM提供了参数来设置分代年龄的大小,但是这个大小不能超过15。
而从设计角度来看,当一个对象触发了最大值15次gc,还没有办法被回收,就只能移动到old generation了。
另外,设计者还引入了动态对象年龄判断的方式来决定把对象转移到old generation,也就是说
不管这个对象的gc年龄是否达到了15次,只要满足动态年龄判断的依据,也同样会转移到old generation。以上就是我对这个问题的理解。
总结
这个问题被问到的频率还挺高的。
并且底层涉及到的知识点也非常多,比如对象头、jvm垃圾回收机制、堆内存划分等等。
所以建议大家在平时工作之外的时间,多花一点时间去研究这些底层原理。
另外,我将所有Java面试系列制作成了完整的面试文档。它的便捷之处在于,可以通过检索的方式,找到你想要的面试题,目前已经更新180期,总计超过15W字!