JVM内存管理<一>:Java内存异常问题排查

一、 内存溢出问题的排查

1. 使用工具 - jdk自带

  • jmap
  • visualvm

2. 流程

  1. 堆转储:
    (1) 方法一:程序运行时,采用:jmap -dump:format=b,file=d:\\data\\xxlJob.hprof 23300 进行堆文件的转储
    (2) 方法二:在内存溢出的时候,打印堆转储文件
java 复制代码
-XX:+HeapDumpOnOutOfMemoryError           # 开启OOM时自动转储堆
-XX:HeapDumpPath=/path/to/dump/dir/       # 指定堆转储文件路径
-XX:OnOutOfMemoryError="command args"     # OOM时执行外部命令
  1. 导入visualVM中
    文件->装入->选择堆转储文件
  1. 分析:自定义的对象的数量、大小都不应该太大
    a. 通过大小、对象数量进行排序,查看靠前的对象是否存在自定义的对象
    b. 通过名称排序,查看自定义的对象中,哪个对象最靠前

二、gc频繁问题的排查

1. 了解堆分配的策略

a. 优先分配到年轻代

b. 大对象直接分配到老年代

c. 长期存活的对象进入老年代

d. 对象担保策略:如果年轻代的对象的总容量大于了老年代的剩余容量,需要进行一次full gc

2. 频繁minor gc

原因

(1) 内存分配过小

(2) 内存泄漏

(3) 频繁的对象创建
排查和解决方案

(1)经过mimor gc之后,对象仍旧无法被清除,导致full gc:考虑内存泄漏问题,查看是否存在内存泄漏,可以通过jmap + visualvm进行排查

(2) 如果是并发高的场景:minor gc之后,考虑eden分配过小,尝试增大内存的分配量,尤其是eden区的

(3) 如果不是高并发,内存还有急速升高的情况,可能是对象创建太过频繁,或者创建了大对象导致的

总结:分配大内存 + 排查堆转储文件,查看哪个自定义对象的占用内存较大,优化代码

3. 频繁full gc

原因

(1)minor gc分配太小,导致频繁minor gc,进而对象年龄急速增加,进入老年代

(2)full gc 分配太小,空间担保机制,使得minor gc之前,需要进行full gc

(3)内存泄漏或者大对象的创建
解决方案:

(1)首先尝试分配大内存

(2)如果不行,通过排查堆转储文件,也就是使用jmap和visualvm查看较大的自定义对象

相关推荐
程序猿大帅2 小时前
别再只当调包侠了:用 Spring AI 落地 Function Calling,我被大模型硬生生砸出了三个大坑
java
程序员晓琪3 小时前
约定大于配置:基于 Java 包名自动生成 API 版本路由的最佳实践
java·spring boot·后端
Flittly3 小时前
【AgentScope Java新手村系列】(11)中断与恢复
java·spring boot·spring
众少成多积小致巨4 小时前
JNI (Java Native Interface) 技术手册中文参考指南
android·java·c++
东坡白菜4 小时前
破局全栈:前端开发的Java入门实战记录—JPA(2)
java·后端
SimonKing10 小时前
艹,维护AI写的代码,我心态崩了......
java·后端·程序员
用户2986985301410 小时前
Java Word 文档样式进阶:段落与文本背景色设置完全指南
java·后端
小bo波1 天前
从"任意文件复制"深挖Java I/O:字符流与字节流的本质抉择
java·nio·io流·后端开发·文件复制
nanxun8862 天前
记一次诡异的 Docker 容器"串包"故障排查
java
用户1563068103512 天前
Day01 | Java 基础(Java SE)
java