除直接调用System.gc外,触发Full GC执行的情况有如下四种:
1.老年代空间不足
老年代空间只有在新生代对象转入及创建为大对象、大数组时才会出现不足现象,当执行Full GC后空间仍然不足,则抛出 java.lang.OutOfMemoryError:Java heap space
2.Permanet Generation 空间满
Permanet Generation 中存放的为一些class信息,当系统中要加载的类、反射的类和调用的方法较多时,Permanet Generation可能会被占满,在未配置为采用CMS GC的情况下会执行Full GC。如果经过Full GC仍然回收不了,那么JVM会跑出 java.lang.OutOfMemoryError:PermGen space
3.CMS GC时出现promotion failed 和 concurrent mode failure
对于采用CMS进行老年代GC的程序,尤其要注意GC日志中是否有 promotion failed 和 concurrent model failure 两种状况,当这两种状况出现时可能会触发Full GC。 promotion failed是在进行MinorGC时,survivor space放不下、对象只能放入老年代,而此时老年代也放不下造成的;concurrent mode failure是在执行CMS GC的过程中同时有对象放入老年代,而此时老年代空间不足造成的。应对措施:增大survivor space、老年代空间或调低触发并发GC的比率,但在JDK5.0+、JDK6.0+的版本中有可能会由于JDK的bug导致CMS在remark完毕后很久才触发sweeping动作。对于这种状况,可通过设置-XX:CMSMaxAbortablePrecleanTime=5(单位ms)来避免
4,统计得到的MinorGC晋升到老年代的平均大小大于老年代的剩余空间
这是一个较为复杂的触发情况,Hotspot为了避免由于新生代对象晋升到老年代导致老年代空间不足的现象,在进行MinorGC时,做了一个判断,如果之前统计所得到的MinorGC晋升到老年代平均大小大于老年代的剩余空间,那么就直接出发Full GC。