1 原因
- 原因一:系统承载高并发请求,或者处理数据量过大,导致Young GC很贫乏,而且每次Young GC过后存活对象太多,内存分配不合理,Survivor区过小,导致对象频繁进入老年代,频繁触发Full GC。
- 原因二:系统一次性加载过多数据进内存,搞出来很多大对象,导致频繁有大对象进入老年带,必然频繁触发Full GC。比如有一些导出的功能,直接select * from xxx,全量导出,导致创建大对象
- 原因三:系统发生了内存泄漏,莫名其妙创建大量的对象,始终无法回收,一直占用在老年代里,必然频繁触发Full GC。
- 原因四:Metaspace(永久代)因为加载类过多触发Full GC。
- 原因五:误调用System.gc()触发Full GC。
2 解决方法
原因一的解决
通过jstat观察GC情况,合理调整新生代内存以及Eden区Survivor区占比。
原因二、三的解决
- 通过jstat观察GC情况。
- 通过jmap导出堆内存快照,然后在jvisualvm或者MAT进行分析。
原因四的解决
一般都是自定义类加载器频繁加载外部类或者JVM动态代理导致的元空间内存不足触发Full GC,可以通过优化代码解决。
原因五解决
通过JVM配置关闭手动GC,-XX:+DisableExplicitGC。