场景应用:Java 出现 Out Of MemoryError(OOM 错误)的原因有哪些?出现 OOM 错误后,怎么解决?

Posted 流楚丶格念

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了场景应用:Java 出现 Out Of MemoryError(OOM 错误)的原因有哪些?出现 OOM 错误后,怎么解决?相关的知识,希望对你有一定的参考价值。

文章目录


OutOf MemoryError 这种错误可以细分为多种不同的错误,每种错误都有
自身的原因和解决办法,如下所示:

超出堆最大空间

java.lang.OutOfMemoryError: Java heap space

错误原因:此 OOM 是由于 JVM 中 heap 的最大值不满足需要。

解决方法:

  1. 调高 heap 的最大值,即-Xmx 的值调大。
  2. 如果你的程序存在内存泄漏,一味的增加 heap 空间也只是推迟该错误出现的时间而已,所以要检查程序是否存在内存泄漏,比如:
    • 尽早释放无用对象的引用。
    • 避免在循环中创建对象。
    • 使用字符串处理时避免使用String,应使用StringBuffer。
    • 尽量少使用静态变量,因为静态变量存放在永久代,基本不参与垃圾回收。

在 GC 时对象过多

java.lang.OutOfMemoryError: GC overhead limit exceeded

错误原因:此 OOM 是由于 JVM 在 GC 时,对象过多,导致内存溢出,建议调整 GC 的策略,在一定比例下开始 GC 而不要使用默认的策略,或者将新代和老代设置合适的大小,需要进行微调存活率。

解决方法:改变 GC 策略,在老代 80%时就是开始 GC,并且将-XX:SurvivorRatio(-XX:SurvivorRatio=8)-XX:NewRatio(-XX:NewRatio=4)设置的更合理。

perm 的最大值不满足需要

java.lang.OutOfMemoryError: Java perm space

错误原因:此 OOM 是由于 JVM 中 perm 的最大值不满足需要。

解决方法:调高 heap 的最大值,即-XX:MaxPermSize 的值调大。
另外,注意一点,Perm 一般是在 JVM 启动时加载类进来,如果是 JVM 运行较长一段时间而不是刚启动后溢出的话,很有可能是由于运行时有类被动态加载进来,此时建议用 CMS 策略中的类卸载配置。如:-XX:+UseConcMarkSweepGC -XX:+CMSClassUnloadingEnabled

操作系统可分配内存不足

java.lang.OutOfMemoryError: unable to create new native thread

错误原因:当 JVM 向 OS 请求创建一个新线程时,而 OS 却由于内存不足无法创建新的 native 线程。

解决方法:如果 JVM 内存调的过大或者可利用率小于 20%,可以建议将 heap 及 perm 的最大值下调,并将线程栈调小,即-Xss 调小,如:-Xss128k。

JVM分配内存大于堆大小

java.lang.OutOfMemoryError: Requested array size exceeds VM limit

错误原因:此类信息表明应用程序(或者被应用程序调用的 APIs)试图分配一个大于堆大小的数组。例如,如果应用程序 new 一个数组对象,大小为512M,但是最大堆大小为 256M,因此 OutOfMemoryError 会抛出,因为数组的大小超过虚拟机的限制。

解决方法:

  1. 首先检查 heap 的-Xmx 是不是设置的过小。
  2. 如果 heap的-Xmx 已经足够大,那么请检查应用程序是不是存在 bug,例如:应用程序可能在计算数组的大小时,存在算法错误,导致数组的 size 很大,从而导致巨大的数组被分配。

native 堆中分配内存失败,并且堆内存可能接近耗尽

java.lang.OutOfMemoryError: request <size> bytes for <reason>. Out of swap space

错误原因:抛出这类错误,是由于从 native 堆中分配内存失败,并且堆内存可能接近耗尽。这类错误可能跟应用程序没有关系,例如下面两种原因也会导致错误的发生:1) 操作系统配置了较小的交换区。2)系统的另外一个进程正在消耗所有的内存。

解决办法:

  1. 检查 os 的 swap 是不是没有设置或者设置的过小。
  2. 检查是否有其他进程在消耗大量的内存,从而导致当前的 JVM 内存不够分配。

注意:虽然有时<reason>部分显示导致 OOM 的原因,但大多数情况下,<reason>显示的是提示分配失败的源模块的名称,所以有必要查看日志文件,如 crash 时的 hs 文件。

以上是关于场景应用:Java 出现 Out Of MemoryError(OOM 错误)的原因有哪些?出现 OOM 错误后,怎么解决?的主要内容,如果未能解决你的问题,请参考以下文章

出现“out of memory” 导致游戏掉线怎么办?

Item is out of date

cf老是出现out of memory

在java开发中遇到string index out of range:4 这是啥原因

Cf老是掉线然后一串英文 out of memory

显卡缓存的问题 out of vedio memory