JVM 分代的概念
堆,才有分代的概念。
堆,才有分代的概念。
堆,才有分代的概念。
分代
JVM 将内存分为大致分为三个区域,1.8 前后有区别
为什么要分代
不分代行不行
不分代完全可以,但是分代的唯一理由是为了优化GC性能。
怎么优化?
如果不进行分代,每次GC都要把整片堆扫一遍来寻找垃圾对象,太慢。分代直接把对象创建到某一个分代区,这样GC就可以针对这一分代区域就行优化,就不费劲了。
年轻代
eden、survivor、from、to
年轻代分三个部分:
- Eden 区,即新生对象区,除大对象以外,新生对象都创建在这里。
- Survivor 区,即幸存区,该区域又分两个空间
s0 和 s1 分别也叫 from 和 to,默认比例8:1。
对象经历的过程
- 新创建对象,分配到Eden区,除大对象特殊处理。
- 在 Eden 区经过一次GC后,如果仍存活,移到 Survivor。
- 在 Survivor 中又经历一次 GC,年龄会增加一岁。
- 年龄到一定程度,移到老年代。
年轻代使用什么GC算法
是复制算法。
为什么选复制算法?
因为:年轻代中的80%以上的对象很快就死亡。
所以,在年轻代的垃圾回收算法使用的是复制算法。即,将内存分为两块,每次中使用其中一块(Eden 和 From),当一块使用完,则将还存活的对象复制到别一块上。复制算法不会产生碎片。
From 和 to 区的复制过程
对上一节的说明,演示复制的过程:
- 在 GC 开始的前,对象只会存在于 Eden区 和 From 区。为什么 From 区会有对象?
- From区有对象很容易理解,开始发生 GC ,Eden 中的对象被复制到了 To 区中。
而 From 中的对象只有两个结果:
- 年龄达到一定值(年龄阈值,可以通过-XX:MaxTenuringThreshold来设置)的对象会被移动到年老代中。
- 没有到达的复制到 To 区中
- 这次 GC 后,Eden 和 From 被清空。此时,To 和 From 互相交换换角色,也就是说 To 变成 GC 前的 From,类推。
到现在清楚了,为什么 From 中也会有对象,因为第一次发生GC时复制过到 To 区后,交换角色而来的。
年轻代相关的JVM参数
-XX:NewSize和-XX:MaxNewSize
: 用于设置年轻代的大小,建议设为整个堆大小的1/3或者1/4,两个值设为一样大。-XX:SurvivorRatio
用于设置Eden和其中一个Survivor的比值,这个值也比较重要。-XX:+PrintTenuringDistribution
这个参数用于显示每次Minor GC时Survivor区中各个年龄段的对象的大小。-XX:InitialTenuringThreshol
和-XX:MaxTenuringThreshold
用于设置晋升到老年代的对象年龄的最小值和最大值,每个对象在坚持过一次Minor GC之后,年龄就加1。
本博客所有文章除特别声明外,均采用 CC BY-NC-SA 4.0 许可协议。转载请注明来自 人话翻译机!
评论