堆,才有分代的概念。
堆,才有分代的概念。
堆,才有分代的概念。

分代

JVM 将内存分为大致分为三个区域,1.8 前后有区别

jvm

为什么要分代

不分代行不行

不分代完全可以,但是分代的唯一理由是为了优化GC性能。
怎么优化?
如果不进行分代,每次GC都要把整片堆扫一遍来寻找垃圾对象,太慢。分代直接把对象创建到某一个分代区,这样GC就可以针对这一分代区域就行优化,就不费劲了。

年轻代

eden、survivor、from、to

年轻代分三个部分:

  1. Eden 区,即新生对象区,除大对象以外,新生对象都创建在这里。
  2. Survivor 区,即幸存区,该区域又分两个空间
    s0 和 s1 分别也叫 from 和 to,默认比例8:1。

young

对象经历的过程

  1. 新创建对象,分配到Eden区,除大对象特殊处理。
  2. 在 Eden 区经过一次GC后,如果仍存活,移到 Survivor。
  3. 在 Survivor 中又经历一次 GC,年龄会增加一岁。
  4. 年龄到一定程度,移到老年代。

年轻代使用什么GC算法

是复制算法。

为什么选复制算法?
因为:年轻代中的80%以上的对象很快就死亡。
所以,在年轻代的垃圾回收算法使用的是复制算法。即,将内存分为两块,每次中使用其中一块(Eden 和 From),当一块使用完,则将还存活的对象复制到别一块上。复制算法不会产生碎片。

复制算法

From 和 to 区的复制过程

对上一节的说明,演示复制的过程:

  1. 在 GC 开始的前,对象只会存在于 Eden区 和 From 区。为什么 From 区会有对象?
  2. From区有对象很容易理解,开始发生 GC ,Eden 中的对象被复制到了 To 区中。

而 From 中的对象只有两个结果:

  1. 年龄达到一定值(年龄阈值,可以通过-XX:MaxTenuringThreshold来设置)的对象会被移动到年老代中。
  2. 没有到达的复制到 To 区中
  3. 这次 GC 后,Eden 和 From 被清空。此时,To 和 From 互相交换换角色,也就是说 To 变成 GC 前的 From,类推。

到现在清楚了,为什么 From 中也会有对象,因为第一次发生GC时复制过到 To 区后,交换角色而来的。

年轻代相关的JVM参数

  1. -XX:NewSize和-XX:MaxNewSize: 用于设置年轻代的大小,建议设为整个堆大小的1/3或者1/4,两个值设为一样大。
  2. -XX:SurvivorRatio
    用于设置Eden和其中一个Survivor的比值,这个值也比较重要。
  3. -XX:+PrintTenuringDistribution
    这个参数用于显示每次Minor GC时Survivor区中各个年龄段的对象的大小。
  4. -XX:InitialTenuringThreshol-XX:MaxTenuringThreshold
    用于设置晋升到老年代的对象年龄的最小值和最大值,每个对象在坚持过一次Minor GC之后,年龄就加1。