JVM出现OOM的八种原因及解决办法

本文源自转载:JVM出现OOM的八种原因及解决办法

目录

一、堆溢出

1.1 原因

1.2解决方法

二、永久代/元空间溢出

2.1 原因

2.2 解决方法

三、GC overhead limit exceeded

3.1 原因

3.2 解决方法

四、方法栈溢出

4.1 原因

4.2 解决方法

五、非常规溢出

5.1 分配超大数组

5.2 swap溢出

5.3 解决方案

六、本地方法溢出


一、堆溢出

这种场景最为常见,报错信息:

java.lang.OutOfMemoryError: Java heap space复制代码

1.1 原因

1、代码中可能存在大对象分配

2、可能存在内存泄露,导致在多次GC之后,还是无法找到一块足够大的内存容纳当前对象。

1.2解决方法

1、检查是否存在大对象的分配,最有可能的是大数组分配

2、通过jmap命令,把堆内存dump下来,使用mat工具分析一下,检查是否存在内存泄露的问题

3、如果没有找到明显的内存泄露,使用 -Xmx 加大堆内存

4、还有一点容易被忽略,检查是否有大量的自定义的 Finalizable 对象,也有可能是框架内部提供的,考虑其存在的必要性

二、永久代/元空间溢出

报错信息:

java.lang.OutOfMemoryError: PermGen spacejava.lang.OutOfMemoryError: Metaspace复制代码

2.1 原因

永久代是 HotSot 虚拟机对方法区的具体实现,存放了被虚拟机加载的类信息、常量、静态变量、JIT编译后的代码等。

JDK8后,元空间替换了永久代,元空间使用的是本地内存,还有其它细节变化:

  • 字符串常量由永久代转移到堆中
  • 和永久代相关的JVM参数已移除

可能原因有如下几种:

1、在Java7之前,频繁的错误使用String.intern()方法

2、运行期间生成了大量的代理类,导致方法区被撑爆,无法卸载

3、应用长时间运行,没有重启

没有重启 JVM 进程一般发生在调试时,如下面 tomcat 官网的一个 FAQ:

Why does the memory usage increase when I redeploy a web application

That is because your web application has a memory leak.

A common issue are “PermGen” memory leaks. They happen because the Classloader (and the Class objects it loaded) cannot be recycled unless some requirements are met (). They are stored in the permanent heap generation by the JVM, and when you redeploy a new class loader is created, which loads another copy of all these classes. This can cause OufOfMemoryErrors eventually.

(*) The requirement is that all classes loaded by this classloader should be able to be gc’ed at the same time.

2.2 解决方法

因为该OOM原因比较简单,解决方法有如下几种:

1、检查是否永久代空间或者元空间设置的过小

2、检查代码中是否存在大量的反射操作

3、dump之后通过mat检查是否存在大量由于反射生成的代理类

4、放大招,重启JVM

三、GC overhead limit exceeded

这个异常比较的罕见,报错信息:

java.lang.OutOfMemoryError:GC overhead limit exceeded复制代码

3.1 原因

这个是JDK6新加的错误类型,一般都是堆太小导致的。Sun 官方对此的定义:超过98%的时间用来做GC并且回收了不到2%的堆内存时会抛出此异常

3.2 解决方法

1、检查项目中是否有大量的死循环或有使用大内存的代码,优化代码。

2、添加参数 -XX:-UseGCOverheadLimit 禁用这个检查,其实这个参数解决不了内存问题,只是把错误的信息延后,最终出现 java.lang.OutOfMemoryError: Java heap space。

3、dump内存,检查是否存在内存泄露,如果没有,加大内存。

四、方法栈溢出

报错信息:

java.lang.OutOfMemoryError : unable to create new native Thread复制代码

4.1 原因

出现这种异常,基本上都是创建的了大量的线程导致的,以前碰到过一次,通过jstack出来一共8000多个线程。

4.2 解决方法

1、通过 -Xss 降低的每个线程栈大小的容量

2、线程总数也受到系统空闲内存和操作系统的限制,检查是否该系统下有此限制:

  • /proc/sys/kernel/pid_max
  • /proc/sys/kernel/thread-max
  • maxuserprocess(ulimit -u)
  • /proc/sys/vm/maxmapcount

五、非常规溢出

下面这些OOM异常,可能大部分的同学都没有碰到过,但还是需要了解一下

5.1 分配超大数组

报错信息 :

java.lang.OutOfMemoryError: Requested array size exceeds VM limit复制代码

这种情况一般是由于不合理的数组分配请求导致的,在为数组分配内存之前,JVM 会执行一项检查。要分配的数组在该平台是否可以寻址(addressable),如果不能寻址(addressable)就会抛出这个错误。

解决方法就是检查你的代码中是否有创建超大数组的地方。

5.2 swap溢出

报错信息 :

java.lang.OutOfMemoryError: Out of swap space复制代码

这种情况一般是操作系统导致的,可能的原因有:

1、swap 分区大小分配不足;

2、其他进程消耗了所有的内存。

5.3 解决方案

1、其它服务进程可以选择性的拆分出去

2、加大swap分区大小,或者加大机器内存大小

六、本地方法溢出

报错信息 :

java.lang.OutOfMemoryError: stack_trace_with_native_method复制代码

本地方法在运行时出现了内存分配失败,和之前的方法栈溢出不同,方法栈溢出发生在 JVM 代码层面,而本地方法溢出发生在JNI代码或本地方法处。

这个异常出现的概率极低,只能通过操作系统本地工具进行诊断,难度有点大,还是放弃为妙。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
JVM(Java虚拟机)中,OOM(Out of Memory)异常通常由以下几个原因引发: 1. 堆内存不足:堆内存是JVM用于分配对象实例的内存区域。如果应用程序创建了大量的对象,并且堆内存无法容纳这些对象,就会导致堆内存不足的OOM异常。 2. 方法区/永久代内存不足:方法区(在Java 8之前称为永久代)用于存储类的元数据、静态变量、常量池等信息。如果应用程序加载了大量的类或者使用了大量的字符串常量,就可能导致方法区/永久代内存不足的OOM异常。 3. 栈内存溢出:栈内存用于保存方法的调用栈和局部变量等信息。如果方法调用层级过深或者方法中使用了大量的局部变量,就可能导致栈内存溢出的OOM异常。 4. 本地方法栈溢出:本地方法栈用于执行本地方法(Native Method)。如果本地方法执行时需要消耗大量的本地内存,而本地内存不足,就可能导致本地方法栈溢出的OOM异常。 5. 直接内存溢出:直接内存是JVM使用的一种特殊的堆外内存,它不受JVM堆内存大小限制。如果应用程序频繁地分配大量的直接内存,而没有及时释放,就可能导致直接内存溢出的OOM异常。 需要注意的是,不同的JVM实现可能在内存管理方面存在差异,因此具体的OOM异常可能会有所不同。此外,可以通过调整JVM的启动参数或优化代码来减少OOM异常的发生。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值