GC overhead limit exceeded问题
一、为什么会产生GC overhead limit exceeded问题
OutOfMemoryError是java.lang.VirtualMachineError的子类;JVM遇到与资源利用有关的问题时,会抛出该错误。更具体地说,当JVM花太多时间执行垃圾回收并且只能回收很少的堆空间时,就会发生该错误。
根据Java文档,默认情况下,如果Java进程花费其时间的98%以上用于GC,并且每次运行中仅回收不到2%的堆,则JVM配置为抛出此错误。换句话说,这意味着我们的应用程序几乎耗尽了所有可用的内存,而垃圾回收器花费了太多时间尝试清理它,并反复失败。
在这种情况下,用户会遇到应用程序极慢的情况。某些操作通常会在几毫秒内完成,因此需要花费更多时间才能完成。这是因为CPU将其全部容量用于垃圾回收,因此无法执行任何其他任务。
二、有什么表现
具体现象一般来说,有以下几点:
-
程序中大量的死循环或有使用大内存的代码;
-
程序中的递归栈的层数太深了,导致内存不足;
-
给服务器的堆内存设置的太小,没有足够的内存可以使用;
-
数据库中的数据表数据太大,由于没有判断带有条件,导致查询全表数据,查询出来的数据量太大,内存溢出;
三、如何解决
解决办法:
1、除了使用-设置堆内存外Xms1g -Xmx2g
,尝试
-XX:+UseG1GC -XX:G1HeapRegionSize=n -XX:MaxGCPauseMillis=m -XX:ParallelGCThreads=n -XX:ConcGCThreads=n
2、优化代码减少 尽可能重用现有对象以节省一些内存
3、检查是否存在
代码使用while循环,不停的new对象,占用了大量的内存。