OutOfMemoryError 内存溢出问题排查

OutOfMemoryError的几种情况如下:

java.lang.OutOfMemoryError: Java heap space
java.lang.OutOfMemoryError: PermGen space
java.lang.OutOfMemoryError: Requested array size exceeds VM limit
java.lang.OutOfMemoryError: request bytes for . Out of swap space
java.lang.OutOfMemoryError: (Native method)


java.lang.OutOfMemoryError: Java heap space

此错误消息不一定意味着内存泄漏。实际上,问题可能与配置问题一样简单。

例如,我负责分析一直产生这种类型的OutOfMemoryError的应用程序。经过一番调查后,我发现罪魁祸首是阵列实例化,因为需要太多的内存;在这种情况下,并不是应用程序的错,而是应用程序服务器依赖于默认的堆太小了。我通过调整解决了这个问题。

在其他情况下,特别是对于长期存在的应用程序,该消息可能表明我们无意中持有对象的引用,从而阻止垃圾收集器清理它们。这时Java语言等同于内存泄漏。 (注意:应用程序调用的API也可能无意中持有对象引用。)

这些 Java堆空间 OOM的另一个潜在来源是使用finalizers。如果类具有finalize方法,则在垃圾收集时该类型的对象不会被回收。而是在垃圾收集之后,稍后对象将排队等待最终确定。在Sun实现中,finalizers由执行。如果finalizers线程无法跟上finalization队列,那么Java堆可能会填满并且可能抛出OOM。

java.lang.OutOfMemoryError: PermGen space

此错误消息表明已满。永久代是存储类和方法对象的堆的区域。如果应用程序加载了大量类,则可能需要使用-XX:MaxPermSize选项增加永久代的大小。

java.lang.String对象也存储在永久代中。 java.lang.String类维护一个字符串池。调用实习方法时,该方法检查池以查看是否存在等效字符串。如果是这样,它由实习方法返回;如果没有,则将字符串添加到池中。更准确地说,java.lang.String.intern方法返回一个字符串的规范表示;结果是对该字符串显示为文字时将返回的同一个类实例的。如果应用程序实例化大量字符串,则可能需要增加永久代的大小。

注意:您可以使用jmap -permgen命令打印与永久生成相关的统计信息,包括有关内部化String实例的信息。

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

此错误表示应用程序(或该应用程序使用的API)尝试分配大于堆大小的数组。例如,如果应用程序尝试分配512MB的数组但最大堆大小为256MB,则将抛出此错误消息的OOM。在大多数情况下,问题是配置问题或应用程序尝试分配海量数组时导致的错误。

java.lang.OutOfMemoryError: request bytes for . Out of swap space

此消息似乎是一个OOM。但是,当本机堆的分配失败并且本机堆可能将被耗尽时,HotSpot VM会抛出此异常。消息中包括失败请求的大小(以字节为单位)以及内存请求的原因。在大多数情况下,是报告分配失败的源模块的名称。

如果抛出此类型的OOM,则可能需要在操作系统上使用故障排除实用程序来进一步诊断问题。在某些情况下,问题甚至可能与应用程序无关。例如,您可能会在以下情况下看到此错误:

操作系统配置的交换空间不足。
系统上的另一个进程是消耗所有可用的内存资源。

由于本机泄漏,应用程序也可能失败(例如,如果某些应用程序或库代码不断分配内存但无法将其释放到操作系统)。

java.lang.OutOfMemoryError: (Native method)

如果您看到此错误消息并且堆栈跟踪的顶部框架是本机方法,则该本机方法遇到分配失败。此消息与上一个消息之间的区别在于,在JNI或本机方法中检测到Java内存分配失败,而不是在Java VM代码中检测到。

如果抛出此类型的OOM,您可能需要在操作系统上使用实用程序来进一步诊断问题。

Application Crash Without OOM

有时,应用程序可能会在从本机堆分配失败后很快崩溃。如果您运行的本机代码不检查内存分配函数返回的错误,则会发生这种情况。

例如,如果没有可用内存,malloc系统调用将返回NULL。如果未检查malloc的返回,则应用程序在尝试访问无效的内存位置时可能会崩溃。根据具体情况,可能很难定位此类问题。

在某些情况下,致命错误日志或崩溃转储的信息就足以诊断问题。如果确定崩溃的原因是某些内存分配中缺少错误处理,那么您必须找到所述分配失败的原因。与任何其他本机堆问题一样,系统可能配置了但交换空间不足,另一个进程可能正在消耗所有可用内存资

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Java内存溢出可能有多种原因,其中一种可能是程序中的内存泄漏,另一种可能是程序中的无限循环。要排查内存溢出,可以使用内存分析工具来查找内存中的问题,也可以检查程序中是否有内存泄漏或无限循环,并采取相应措施来解决这些问题。 ### 回答2: Java内存溢出是指程序在运行过程中申请的内存超过了Java虚拟机所能提供的最大内存限制,导致程序异常终止或出现错误。为了排查Java内存溢出问题,可以采取以下几个步骤: 1. 检查错误信息:查看程序的错误日志或控制台输出,通常会提示内存溢出的错误信息。错误信息可能包括“java.lang.OutOfMemoryError”等相关信息,能够帮助我们定位问题。 2. 分析堆栈信息:出现内存溢出时,通常会在堆栈信息中找到一些线索。查看堆栈信息,找出哪些方法或代码片段出现频繁调用,占用大量内存的可能性较大。 3. 使用工具分析内存:使用一些Java内存分析工具,如Eclipse Memory Analyzer(MAT)、VisualVM等,可以帮助我们定位问题。这些工具能够生成内存快照,并提供一个可视化界面来查看内存使用情况、对象引用关系等。通过分析内存快照,我们可以找到内存泄漏或过多使用内存的问题。 4. 代码审查:检查程序的代码,查看是否有明显的内存泄漏问题。例如未关闭数据库连接、未释放资源等。 5. 调整堆内存大小:可以通过增大Java虚拟机的堆内存限制来缓解内存溢出问题。可以通过设置JVM选项-Xmx和-Xms来增大最大堆和初始堆的大小。 总之,Java内存溢出问题排查需要结合错误信息、堆栈信息、内存分析工具等多种方法,并进行深入分析和定位。通过找出导致内存溢出的具体原因,以便采取相应的措施解决问题。 ### 回答3: Java内存溢出是指在程序运行过程中,因为申请的内存超过了JVM堆内存的限制,导致程序抛出OutOfMemoryError异常的情况。接下来我们将介绍一些排查内存溢出的常用方法。 第一步是分析错误日志。当程序抛出内存溢出异常时,JVM会生成错误日志文件,其中包含了异常的堆栈信息以及可能的原因。通过分析错误日志,我们可以初步了解是哪一部分代码导致了内存溢出。 第二步是使用内存分析工具。常用的内存分析工具有VisualVM、Eclipse Memory Analyzer等。这些工具可以帮助我们查看内存使用情况、对象的引用关系以及内存泄漏等信息。通过分析内存快照,我们可以找到内存溢出的具体原因。 第三步是检查代码中的资源泄漏问题。在Java程序中,未正确关闭的资源(如文件流、数据库连接等)会导致内存泄漏。因此,我们需要仔细检查代码,确保使用完资源后及时关闭。 第四步是调整JVM参数。通过调整JVM参数,可以增加JVM堆内存的大小或优化垃圾回收机制,以减少内存溢出的风险。常用的参数有-Xmx、-Xms和-XX:MaxPermSize等。 最后一步是代码优化。如果经过以上步骤仍然无法解决内存溢出问题,那么可能是代码设计存在问题。我们需要检查代码中是否有频繁创建大量对象的情况,是否有不必要的数据缓存等,以优化代码并降低内存消耗。 总之,解决Java内存溢出问题需要结合分析错误日志、使用内存分析工具、检查资源泄漏、调整JVM参数和代码优化等多个步骤,以找到并解决问题的根源。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值