当前位置: 首页 > article >正文

JVM 内存溢出 详解

内存溢出

内存溢出指的是内存中某一块区域的使用量超过了允许使用的最大值,从而使用内存时因空间不足而失败,虚拟机一般会抛出指定的错误。

在Java虚拟机中,只有程序计数器不会出现内存溢出的情况,因为每个线程的程序计数器只保存一个固定长度的地址。

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

堆内存溢出

java.lang.OutOfMemoryError: Java heap space

​ 堆内存溢出指的是在堆上分配的对象空间超过了堆的最大大小,从而导致的内存溢出。堆的最大大小使用-Xmx参数进行设置,如-Xmx10m代表最大堆内存大小为10m。

​ Java堆用于储存对象实例,我们只要不断地创建对象,并且保证GC Roots到对象之间有可达路径 来避免垃圾回收机制清除这些对象,那么随着对象数量的增加,总容量触及最大堆的容量限制后就会 产生内存溢出异常。

栈内存溢出

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

​ 栈内存溢出指的是所有栈帧空间的占用内存超过了最大值,最大值使用-Xss进行设置,比如-Xss256k代表所有栈帧占用内存大小加起来不能超过256k。

方法区内存溢出

java.lang.OutOfMemoryError: Metaspace / PermGen space

方法区内存溢出指的是方法区中存放的内容比如类的元信息超过了方法区内存的最大值,JDK7及之前版本方法区使用永久代(-XX:MaxPermSize=值)来实现,JDK8及之后使用元空间(-XX:MaxMetaspaceSize=值)来实现。

原因:加载的类过多或动态生成类(如反射、CGLIB),超出元空间限制(-XX:MaxMetaspaceSize)。

import javassist.ClassPool;public class MetaspaceOOM {public static void main(String[] args) throws Exception {ClassPool cp = ClassPool.getDefault();for (int i = 0; i < 100000; i++) {// 动态生成类Class<?> clazz = cp.makeClass("MetaspaceOOM" + i).toClass();}}
}

直接内存溢出

java.lang.OutOfMemoryError: Direct buffer memory

​ 直接内存溢出指的是申请的直接内存空间大小超过了最大值,使用 -XX:MaxDirectMemorySize=值 设置最大值。溢出之后会抛出OutOfMemoryError:

总结:

OOM 类型触发原因解决方案
堆内存溢出对象过多/内存泄漏调整 -Xmx,优化代码
元空间溢出动态类过多限制 -XX:MaxMetaspaceSize
栈溢出(线程数过多)线程数超出系统限制使用线程池,调整 -Xss
直接内存溢出NIO 分配过多直接内存调整 -XX:MaxDirectMemorySize
  1. 通过 jstatjmapjstack 监控内存使用。
  2. 生成堆转储文件(-XX:+HeapDumpOnOutOfMemoryError),用 MAT 分析内存泄漏。
  3. 避免过度依赖反射、动态代理等易触发元空间问题的技术。

如何避免 OOM?

1. 堆内存溢出
• 诊断工具:

​ • 使用 jvisualvmMAT(Memory Analyzer Tool)分析堆转储(-XX:+HeapDumpOnOutOfMemoryError)。

​ • 检查是否有内存泄漏(对象被意外长期引用)。

​ • 优化方法:

​ • 调整堆大小(-Xmx-Xms)。

​ • 优化代码,及时释放无用对象(如清理集合、关闭资源)。

​ • 避免创建超大对象(如大数组)。

2. 元空间溢出
• 诊断工具:

​ • 使用 jstat -gcmetacapacity 监控元空间使用情况。

​ • 检查动态生成类的代码(如反射、动态代理)。

​ • 优化方法:

​ • 限制元空间大小(-XX:MaxMetaspaceSize=256m)。

​ • 减少动态类生成(如缓存反射生成的类)。

3. 栈溢出(线程数过多)
• 诊断工具:

​ • 检查线程数(ps -eLf | grep java)。

​ • 分析线程栈(jstack)。

​ • 优化方法:

​ • 减少线程数(使用线程池)。

​ • 调整线程栈大小(-Xss256k)。

4. 直接内存溢出
• 诊断工具:

​ • 监控 java.nio.BitsreservedMemory(NIO 内存使用)。

​ • 优化方法:

​ • 显式释放直接内存(调用 ((DirectBuffer) buffer).cleaner().clean())。

​ • 调整 -XX:MaxDirectMemorySize

http://www.lryc.cn/news/2403585.html

相关文章:

  • Qt 5.12 上读取 .xlsx 文件(Windows 平台)
  • 虚拟机CentOS 7 网络连接显示“以太网(ens33,被拔出)“、有线已拔出、CentOS7不显示网络图标
  • Tailwind CSS 实战:基于 Kooboo 构建 AI 对话框页面(六):图片上传交互功能
  • 传统的将自然语言转化为嵌入向量的核心机制是:,将离散的语言符号转化为连续的语义向量,其核心依赖“上下文决定语义”的假设和神经网络的特征提取能力。
  • 【前端】每日一道面试题6:解释Promise.any和Promise.allSettled的使用场景及区别。
  • wordpress+woocommerce电商平台搭建方案的优势分析
  • 玄机-日志分析-IIS日志分析
  • IDEA:配置 Git 需要完成 Git 路径设置、账号认证以及仓库关联三个主要步骤
  • PHP 复制商品扩展实操:轻松切换一号通、99api ,实现商品复制功能
  • 【办公类-104-01】20250606通义万相50分一天用完,通义万相2.1专业版测试
  • Prompt Engineering Notes
  • C++课设:学生成绩管理系统
  • 制作个人Github学术主页
  • 【Linux内核】设备模型之udev技术详解
  • FineReport模板认证找不到模板
  • STM32实战:数字音频播放器开发指南
  • 豆包和deepseek 元宝 百度ai区别是什么
  • TomatoSCI数据分析实战:探索社交媒体成瘾
  • 网络安全厂商F5推出AI Gateway,化解大模型应用风险
  • pikachu靶场通关笔记16 CSRF关卡02-CSRF(POST)
  • 场景题-3
  • Java 类型参数 T、R 、 O 、K、V 、E 、? 区别
  • 中医的十问歌和脉象分类
  • C#封装HttpClient:HTTP请求处理最佳实践
  • 前端基础之《Vue(19)—状态管理》
  • 构建 MCP 服务器:第 4 部分 — 创建工具
  • 2.1 Windows编译环境介绍
  • 如何以 9 种方式将照片从手机传输到笔记本电脑
  • 生成JavaDoc文档
  • 八股学习-JS的闭包