GC的介绍
-
Partial GC:部分收集,不是对整个Java堆和方法区的收集,其中包括以下三种GC。
- Minor GC/Young GC:新生代收集,只对新生区(Eden+S0/S1)的收集
- Major GC/Old GC:老年代收集,只对老年代进行收集(只有CMS的concurrent collection是这个模式),但有时候材料提到的Major GC跟Full GC一致功能(整堆收集),这里我觉得它应该是只对老年代进行收集。
- Mixed GC:混合收集,对整个新生代以及部分老年代的收集,目前只有G1收集器能实现
-
Full GC:整堆收集,对整个Java堆(新生代,老年代)和方法区(永久区/元空间)的收集
Minor GC/Young GC:
- 只有在新生代中的Eden区满,申请对象时才会发生新生代收集。
- 因为对象绝大部分都在新生代中"朝声夕灭",所以在新生代中发生的GC频率很高,一般这个GC的速度是比较快的。
- 此收集也会引起STW(STOP THE WORLD),会暂停除了GC以外的所有线程,只有等垃圾回收结束,用户线程才能恢复运行。
Major GC/Old GC:
- 每次Major GC的出现,经常伴随着最少一次的Minor GC,但不是绝对的,在Parallel Scavenge收集器的收集策略就有直接进行Major GC的策略选择过程。
- 当老年代空间不足的时候,会尝试的发生Minor GC,若之后空间还是不够则会出发Major GC。
- Major GC的速度会比Minor GC满10倍以上,也就是STW的时间会更长。
- 如果进行Major GC后,内存还是不足,就会报OOM异常。
Full GC:
- 当用户自己调用System.gc()的方法时,可能会执行Full GC。
- 当老年代空间或者方法区空间不足的时候。
- 这里的老年代空间不足不一定会调用Full GC,若老年代收集器选择CMS收集器的话,则会发生Major GC。
- Full GC是开发中调优尽量要避免的。
案例:
public class GCTest {
public static void main(String[] args) {
int i = 0;
try {
List<String> list = new ArrayList<>();
String a = "GCTest";
while (true) {
list.add(a);
a = a + a;
i++;
}
} catch (Throwable t) {
t.printStackTrace();
System.out.println("遍历次数为:" + i);
}
}
}
对以上代码进行测试,在VM参数中设置"-Xms10m -Xmx10m -XX:+PrintGCDetails"