在Java的垃圾回收(Garbage Collection)过程中,存在三种不同的垃圾回收类型:Minor GC(新生代垃圾回收)、Major GC(老年代垃圾回收)和Full GC(完全垃圾回收)。它们有不同的功能和触发条件:
Minor GC(新生代垃圾回收)
- 触发条件:当新生代空间不足时,会触发Minor GC。新生代主要包括一个Eden区和两个Survivor区(通常是一个From区和一个To区)。
- 执行过程:在Minor GC中,只清理新生代内存空间。首先,将存活的对象从Eden区和Survivor区复制到另外一个Survivor区,然后清空原有的Eden区和From区。这样,在每次Minor GC后,都会有部分对象存活并被复制到另外一个Survivor区。
- 目的:Minor GC的目标是清理新生代空间中的无用对象,并将存活对象复制到Survivor区或老年代。通常,新生代中的对象生命周期较短,因此Minor GC会频繁执行。
Major GC(老年代垃圾回收)
- 触发条件:老年代会在两种情况下触发 Old GC:
- 一是开启分配担保机制,根据历次 Minor GC 后进入老年代的对象大于当前老年代内存大小,判断 Minor GC 有风险,则会触发 Old GC;
- 二是 Minor GC 后剩余对象太多,老年代放不下了也会触发 Old GC。
- 执行过程:Major GC执行时,会对整个堆(包括新生代和老年代)进行垃圾回收。它会暂停应用程序的执行,清理整个堆中的无用对象。
- 目的:Major GC的目标是清理堆中的无用对象,尤其是老年代中的对象。由于老年代中的对象生命周期较长,Major GC相对较少发生。
Full GC(完全垃圾回收)
FullGC会产生STW(Stop The Wolrd)即为了保证对象引用更新的正确性,必须暂停所有用户线程,所以必须尽量避免发生FullGC
在MinorGC之前,会先检查老年代中最大可用连续空间是否大于新生代所有对象的总空间
- 如果小于,说明MinorGC是不安全的,则会查看参数 HandlePromotionFailure 是否被设置成了允许担保失败,如果不允许则直接触发Full GC;
- 如果允许,那么会进一步检查老年代最大可用的连续空间是否大于历次晋升到老年代对象的平均大小,如果小于也会触发 Full GC
元空间满时会触发FullGC,可以通过-XX:MetaspaceSize 参数来设置元空间的大小
手动调用System.gc()方法,但并不保证一定会触发Full GC。但大概率会增加FullGC的概率一般建议不要手动进行此方法的调用,可以通过
-XX:+ DisableExplicitGC
来禁止RMI调用System.gc