JVM堆区新生代为什么有两个Survivor

9 篇文章 0 订阅

原文:https://www.jianshu.com/p/3d3fc356e31c 

JVM内存模型中, Heap区被分为新生代和老年代两个区域, 新生代又分为Eden+Survivor1+Survivor2. 新生代收集算法主要使用复制算法, 老年代收集算法主要使用标记-清理或标记-整理算法.

堆内存划分

 

新生代在发生首次YGC的时候, Eden内存活的对象会被复制到S1.

 

再发生YGC的时候, Eden内存活的对象和S1内存活的对象会被复制到S2, 同时清除Eden内的对象和S1内的对象.

 

再发生YGC的时候, Eden内存活的对象和S2内存活的对象会被复制到S1, 同时清除Eden内的对象和S2内的对象.

 

以此往复循环. 生存次数超过阈值的对象进入老年代. 可以总结得出: 每次YGC发生之后, S1和S2总会有一个是空的, 这样子的目的是避免内存碎片化带来的空间与性能损失. 请注意笔者画图的时候故意画出间隔很大的不同内存区域, 实际情况中这就是存活对象的内存分布.
现在想象一下如果只有一个Survivor区, 那么每次YGC的时候, Eden区和S区都会有内存碎片, 这是无疑的一点, 此时如果将Eden区内存活的对象直接复制到S区, 那么内存情况将会是如下:
(惨不忍睹...)


随着一次次的YGC, S区内的内存碎片无疑会变得越来越多.内存浪费非常严重. 这里有人可能会想: 我们好像也可以在每次YGC的时候, 对S区内的对象进行重排列, 使得S区内的对象一个个紧挨着彼此, 避免内存碎片化. 这个想法是可以的, 但是要考虑到YGC是JVM垃圾收集中最最最频繁的活动. 如果每次YGC时都要花费这么多时间去重排列对象, 对象重排列相比直接复制, 是很耗时的计算. 因此划分出两个Survivor区域, 以空间为代价(每次都有一个S区为空)换取GC时间, 是很值得的事情. 提升了服务器响应性.

作者:OisCircle
链接:https://www.jianshu.com/p/3d3fc356e31c
来源:简书
简书著作权归作者所有,任何形式的转载都请联系作者获得授权并注明出处。

  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值