Java虚拟机OOM解析

[注:本文内容摘选自作者之前的博客]

对于以Java为核心的企业级应用开发,Java虚拟机(JVM)是项目架构的核心。只有深入理解其内存分配和垃圾回收机制,才能在项目建设中得心应手,无论是基于当前流行的微服务架构(如Spring Cloud或Dubbo),还是即将(或已经)流行的服务网格体系。

不多废话,先看一张Java虚拟机的内存体系结构图:

Java虚拟机OOM解析 注:本文主要基于Oracle JDK体系进行阐述

此图主要针对Java Lang Out Of Memory Error异常进行了适当调整,因为核心问题和要素可以通过此图展示。本文主要解析OOM异常,若想深入了解Java虚拟机的相关知识,可参考官方文档。

立即学习“Java免费学习笔记(深入)”;

当我们的应用程序因Java虚拟机内存分配异常时,会抛出java.lang.OutOfMemoryError错误。以下结合实际项目经验,总结了常见的OOM原因及其解决方法,如有遗漏或错误,欢迎补充指正。

1、Java Heap Space

此情况表明当堆内存(Heap Space)没有足够空间存放新创建的对象时,会抛出java.lang.OutOfMemoryError: Java heap space错误。

原因分析

Java heap space错误的常见原因包括以下几类:

1、请求创建一个超大对象,通常是一个大数组

2、超出预期的访问量/数据量,通常是上游系统请求流量飙升

3、过度使用终结器(Finalizer),该对象没有立即被GC

4、内存泄漏(Memory Leak),大量对象引用没有释放,JVM无法自动回收,常见于使用File等资源未回收

解决方案

针对大多数情况,通常只需通过-Xmx参数增加JVM堆内存空间即可。如果问题仍然存在,可以参考以下情况进行进一步处理:

1、如果是超大对象,检查其合理性,例如是否一次性查询了数据库全部结果,而没有限制结果数量。

2、如果是业务峰值压力,可以考虑增加机器资源或进行限流降级。

3、如果是内存泄漏,需要找到持有的对象,优化代码设计。

2、GC overhead limit exceeded

此情况为JDK1.6新增的错误类型。如果没有此异常,会发生什么?垃圾回收释放的2%可用内存空间会迅速被填满,迫使GC再次执行,导致频繁执行GC操作,服务器会因为频繁的GC操作达到100%的使用率,运行变慢,应用系统会出现卡顿现象,通常几毫秒可完成的操作,现在需要更长时间,甚至几分钟才能完成。

原因分析

1、垃圾回收可用空间较小

2、频繁执行GC操作,使得资源使用率较高

解决方案

1、增加heap堆内存

2、增加内存后错误依旧,获取heap内存快照,使用Eclipse MAT工具,找出内存泄露并进行修复

3、优化代码以使用更少的内存或重用对象,而不是创建新的对象,从而减少垃圾收集器的运行次数。如果代码中在循环中创建了许多临时对象,应该尝试重用它们

4、升级JDK到1.8,至少也是1.7,并使用G1 GC垃圾回收算法

5、优化代码

3、Unable to create new native thread

每个Java线程都需要占用一定的内存空间,当JVM向底层操作系统请求创建一个新的native线程时,如果没有足够的资源分配就会报此类错误。

原因分析

JVM向OS请求创建native线程失败,就会抛出Unable to create new native thread错误,常见原因包括以下几类:

1、线程数超过操作系统最大线程数ulimit限制

2、线程数超过kernel.pid_max(只能重启)

3、native内存不足

解决方案

1、升级配置,为机器提供更多的内存

2、降低Java Heap Space大小

3、修复应用程序的线程泄漏问题

4、限制线程池大小

5、使用-Xss参数减少线程栈的大小

6、调高OS层面的线程最大数:执行ulimit -a查看最大线程数限制,使用ulimit -u xxx调整最大线程数限制

4、Out of swap space?

此错误表示所有可用的虚拟内存已被耗尽。虚拟内存(Virtual Memory)由物理内存(Physical Memory)和交换空间(Swap Space)组成。当程序运行时请求的虚拟内存溢出时,会报Out of swap space错误。

原因分析

此错误出现的常见原因包括以下几类:

1、地址空间不足

2、物理内存已耗光

百度虚拟主播 百度虚拟主播

百度智能云平台的一站式、灵活化的虚拟主播直播解决方案

百度虚拟主播 36 查看详情 百度虚拟主播

3、应用程序的本地内存泄漏(native leak),例如不断申请本地内存却不释放

4、执行jmap -histo:live 命令,强制执行Full GC;如果几次执行后内存明显下降,则基本确认为Direct ByteBuffer问题

解决方案

根据错误原因可以采取如下解决方案:

1、升级地址空间为64 bit

2、使用Arthas检查是否为Inflater/Deflater解压缩问题,如果是,则显式调用end方法

3、Direct ByteBuffer问题可以通过启动参数-XX:MaxDirectMemorySize调低阈值

4、升级服务器配置/隔离部署,避免争用

5、Kill process or sacrifice child

有一种内核作业(Kernel Job)称为Out of Memory Killer,它会在可用内存极低的情况下“杀死”(kill)某些进程。OOM Killer会对所有进程进行评分,然后将评分较低的进程“杀死”,具体的评分规则可以参考Surviving the Linux OOM Killer。

不同于其他的OOM错误,Kill process or sacrifice child错误不是由JVM层面触发的,而是由操作系统层面触发的。

原因分析

1、默认情况下,Linux内核允许进程申请的内存总量大于系统可用内存,通过这种“错峰复用”的方式可以更有效地利用系统资源

2、然而,这种方式也会无可避免地带来一定的“超卖”风险。例如某些进程持续占用系统内存,然后导致其他进程没有可用内存。此时,系统将自动激活OOM Killer,寻找评分低的进程,并将其“杀死”,释放内存资源

解决方案

1、升级服务器配置/隔离部署,避免争用

2、OOM Killer调优

6、Requested array size exceeds VM limit

JVM限制了数组的最大长度,此错误表示程序请求创建的数组超过最大长度限制。JVM在为数组分配内存前,会检查要分配的数据结构在系统中是否可寻址,通常为Integer.MAX_VALUE-2。

此类问题比较罕见,通常需要检查代码,确认业务是否需要创建如此大的数组,是否可以拆分为多个块,分批执行。

7、Direct buffer memory

Java允许应用程序通过Direct ByteBuffer直接访问堆外内存,许多高性能程序通过Direct ByteBuffer结合内存映射文件(Memory Mapped File)实现高速IO。

原因分析

Direct ByteBuffer的默认大小为64 MB,一旦使用超出限制,就会抛出Direct buffer memory错误。

解决方案

1、Java只能通过ByteBuffer.allocateDirect方法使用Direct ByteBuffer,因此,可以通过Arthas等在线诊断工具拦截该方法进行排查

2、检查是否直接或间接使用了NIO,如netty,jetty等

3、通过启动参数-XX:MaxDirectMemorySize调整Direct ByteBuffer的上限值

4、检查JVM参数是否有-XX:+DisableExplicitGC选项,如果有就去掉,因为该参数会使System.gc()失效

5、检查堆外内存使用代码,确认是否存在内存泄漏;或者通过反射调用sun.misc.Cleaner的clean()方法来主动释放被Direct ByteBuffer持有的内存空间

6、内存容量确实不足,升级配置

8、Permgen space

此错误表示永久代(Permanent Generation)已用满,通常是因为加载的class数目太多或体积太大。

原因分析

永久代存储对象主要包括以下几类:

1、加载/缓存到内存中的class定义,包括类的名称,字段,方法和字节码

2、常量池

3、对象数组/类型数组所关联的class

4、JIT编译器优化后的class信息[PermGen的使用量与加载到内存的class的数量/大小正相关]

解决方案

根据Permgen space报错的时机,可以采用不同的解决方案,如下所示:

1、程序启动报错,修改-XX:MaxPermSize启动参数,调大永久代空间

2、应用重新部署时报错,很可能是应用没有重启,导致加载了多份class信息,只需重启JVM即可解决

3、运行时报错,应用程序可能会动态创建大量class,而这些class的生命周期很短暂,但是JVM默认不会卸载class,可以设置-XX:+CMSClassUnloadingEnabled和-XX:+UseConcMarkSweepGC这两个参数允许JVM卸载class

9、Metaspace

原因分析

JDK1.8使用Metaspace替换了永久代(Permanent Generation),此错误表示Metaspace已被用满,通常是因为加载的class数目太多或体积太大。

解决方案

此类问题的原因与解决方法跟Permgen space非常类似,可以参考上文。需要特别注意的是调整Metaspace空间大小的启动参数为-XX:MaxMetaspaceSize。

以上就是Java虚拟机OOM解析的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/484708.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月8日 12:50:31
下一篇 2025年11月8日 12:51:34

相关推荐

  • Bear 博客上的浅色/深色模式分步指南

    我最近使用偏好颜色方案媒体功能与 light-dark() 颜色函数相结合,在我的 bear 博客上实现了亮/暗模式切换。 我是这样做的。 第 1 步:设置 css css 在过去几年中获得了一些很酷的新功能,包括 light-dark() 颜色函数。此功能可让您为任何元素指定两种颜色 &#8211…

    2025年12月24日
    100
  • 如何在 Web 开发中检测浏览器中的操作系统暗模式?

    检测浏览器中的操作系统暗模式 在 web 开发中,用户界面适应操作系统(os)的暗模式设置变得越来越重要。本文将重点介绍检测浏览器中 os 暗模式的方法,从而使网站能够针对不同模式调整其设计。 w3c media queries level 5 最新的 web 标准引入了 prefers-color…

    2025年12月24日
    000
  • 如何使用 CSS 检测操作系统是否处于暗模式?

    如何在浏览器中检测操作系统是否处于暗模式? 新发布的 os x 暗模式提供了在 mac 电脑上使用更具沉浸感的用户界面,但我们很多人都想知道如何在浏览器中检测这种设置。 新标准 检测操作系统暗模式的解决方案出现在 w3c media queries level 5 中的最新标准中: 立即学习“前端免…

    2025年12月24日
    000
  • 如何检测浏览器环境中的操作系统暗模式?

    浏览器环境中的操作系统暗模式检测 在如今科技的海洋中,越来越多的设备和软件支持暗模式,以减少对眼睛的刺激并营造更舒适的视觉体验。然而,在浏览器环境中检测操作系统是否处于暗模式却是一个令人好奇的问题。 检测暗模式的标准 要检测操作系统在浏览器中是否处于暗模式,web 开发人员可以使用 w3c 的媒体查…

    2025年12月24日
    200
  • 浏览器中如何检测操作系统的暗模式设置?

    浏览器中的操作系统暗模式检测 近年来,随着用户对夜间浏览体验的偏好不断提高,操作系统已开始引入暗模式功能。作为一名 web 开发人员,您可能想知道如何检测浏览器中操作系统的暗模式状态,以相应地调整您网站的设计。 新 media queries 水平 w3c 的 media queries level…

    2025年12月24日
    000
  • 如何在 VS Code 中解决折叠代码复制问题?

    解决 VS Code 折叠代码复制问题 在 VS Code 中使用折叠功能可以帮助组织长代码,但使用复制功能时,可能会遇到只复制可见部分的问题。以下是如何解决此问题: 当代码被折叠时,可以使用以下简单操作复制整个折叠代码: 按下 Ctrl + C (Windows/Linux) 或 Cmd + C …

    2025年12月24日
    000
  • 我在学习编程的第一周学到的工具

    作为一个刚刚完成中学教育的女孩和一个精通技术并热衷于解决问题的人,几周前我开始了我的编程之旅。我的名字是OKESANJO FATHIA OPEYEMI。我很高兴能分享我在编码世界中的经验和发现。拥有计算机科学背景的我一直对编程提供的无限可能性着迷。在这篇文章中,我将反思我在学习编程的第一周中获得的关…

    2025年12月24日
    000
  • 姜戈顺风

    本教程演示如何在新项目中从头开始配置 django 和 tailwindcss。 django 设置 创建一个名为 .venv 的新虚拟环境。 # windows$ python -m venv .venv$ .venvscriptsactivate.ps1(.venv) $# macos/linu…

    2025年12月24日
    000
  • 黏性定位的失效原因及解决方法

    粘性定位为什么会失效?原因及解决方法 一、引言在前端开发中,粘性定位(sticky position)是一种常见的布局方式。通过设置元素的定位属性为sticky,可以实现在指定的滚动范围内,元素在页面上的位置保持固定不变,直到达到指定的偏移量。然而,有时候我们会发现粘性定位失效的情况,本文将探讨其原…

    2025年12月24日
    000
  • 分析与解决绝对定位故障的原因

    绝对定位故障的原因分析及解决方法 概述:绝对定位是前端开发中常见的一种布局方式,它可以让元素在页面中精确地定位。但是,在实际的开发过程中,我们可能会遇到绝对定位出现故障的情况。本文将分析绝对定位故障的原因,并提供解决方法,同时附上具体的代码示例。 一、原因分析: 定位元素和参照元素的父元素未设置定位…

    2025年12月24日
    000
  • CSS主框架偏移的原因及解决方法推导

    解析CSS主框架偏移的原因及解决方法,需要具体代码示例 标题:CSS主框架偏移问题的分析与解决方案 引言:随着Web开发的不断发展,CSS作为前端开发的重要工具之一,被广泛应用于页面布局和样式设计。然而,在实际开发中,我们可能会遇到CSS主框架偏移的问题,即页面元素无法按预期位置显示。本文将深入分析…

    2025年12月24日
    200
  • CSS中IE浏览器最基本的一些bug以及解决方法

    css如何解决bug?相信有很多刚刚接触css中ie浏览器的朋友都会有这样的疑问。本章就给大家介绍css中ie浏览器最基本的一些bug以及解决方法。有一定的参考价值,有需要的朋友可以参考一下,希望对你们有所帮助。 一、IE6双倍边距bug 当页面上的元素使用float浮动时,不管是向左还是向右浮动;…

    2025年12月24日
    300
  • jimdo能否添加html5弹窗_jimdo弹窗html5代码实现与触发条件【技巧】

    可在Jimdo实现HTML5弹窗的四种方法:一、用内置“弹窗链接”模块;二、通过HTML区块注入精简dialog结构(需配合内联CSS);三、外部托管HTML+iframe嵌入;四、纯CSS :target伪类无JS方案。 如果您希望在Jimdo网站中实现HTML5弹窗效果,但发现平台默认不支持直接…

    2025年12月23日
    000
  • 响应式HTML5按钮适配不同屏幕方法【方法】

    实现响应式HTML5按钮需五种方法:一、CSS媒体查询按max-width断点调整样式;二、用rem/vw等相对单位替代px;三、Flexbox控制容器与按钮伸缩;四、CSS变量配合requestAnimationFrame优化的JS动态适配;五、Tailwind等框架的响应式工具类。 如果您希望H…

    2025年12月23日
    000
  • jimdo如何添加html5表单_jimdo表单html5代码嵌入与字段设置【实操】

    可通过嵌入HTML5表单代码、启用字段验证属性、添加CSS样式反馈及替换提交按钮并绑定JS事件四种方式在Jimdo实现自定义表单行为。 如果您在 Jimdo 网站中需要自定义表单行为或字段逻辑,而内置表单编辑器无法满足需求,则可通过嵌入 HTML5 表单代码实现更灵活的控制。以下是具体操作步骤: 一…

    2025年12月23日
    000
  • vs里面怎么html5_VS新建项目选HTML5模板或文件选HTML5创建【创建】

    Visual Studio 中创建 HTML5 项目可通过四种方式:一、新建空 ASP.NET Web 应用程序后添加 HTML 页面;二、使用 UWP 的 Blank App 模板;三、直接新建 HTML 文件并手动编写标准 HTML5 结构;四、安装 Web Template Studio 扩展…

    2025年12月23日
    000
  • html5能否禁用搜索框自动填充_html5autocomplete关闭方法【教程】

    禁用HTML5搜索框自动填充有五种方法:一、设autocomplete=”off”;二、随机化name/id值;三、用无效autocomplete值如”nope”;四、JS动态设置autocomplete;五、设autocomplete=”…

    2025年12月23日
    000
  • 如何查看编写的html_查看自己编写的HTML文件效果【效果】

    要查看HTML文件的浏览器渲染效果,需确保文件以.html为扩展名保存、用浏览器直接打开、利用开发者工具调试、必要时启用本地HTTP服务器、或使用编辑器实时预览插件。 如果您编写了HTML代码,但无法直观看到其在浏览器中的实际渲染效果,则可能是由于文件未正确保存、未使用浏览器打开或文件扩展名设置错误…

    2025年12月23日
    400
  • html5怎么加php_html5用Ajax与PHP后端交互实现数据传递【交互】

    HTML5不能直接运行PHP,需通过Ajax与PHP通信:前端用fetch发送请求,PHP接收处理并返回JSON,前端解析响应更新DOM;注意跨域、编码、CSRF防护和输入过滤。 HTML5 本身是前端标记语言,不能直接运行 PHP 代码,但可以通过 Ajax(异步 JavaScript)与 PHP…

    2025年12月23日
    300
  • html5怎么设置单选_html5用input type=”radio”加name设单选按钮组【设置】

    HTML5 使用 type=”radio” 实现单选功能,需统一 name 值构成互斥组;通过 checked 设默认项;可用 CSS 隐藏原生控件并自定义样式;推荐用 fieldset/legend 增强语义;required 可实现必填验证。 如果您希望在网页中创建一组互…

    2025年12月23日
    200

发表回复

登录后才能评论
关注微信