【JVM】如何定位、解决内存泄漏和溢出

简介: 【JVM】如何定位、解决内存泄漏和溢出

1.概述

常见的几种JVM内存溢出的场景如下:

Java堆溢出: 错误信息: java.lang.OutOfMemoryError: Java heap space 原因:Java对象实例在运行时持续创建,但不再使用的对象没有及时被垃圾回收器回收,导致堆内存耗尽。 解决方案:增加堆内存大小(-Xms和-Xmx参数),优化对象生命周期管理,减少不必要的大对象或者长时间存在的临时对象。


永久代/元空间溢出(取决于Java版本): 在Java 8及以前版本中,永久代存储类信息、常量池、静态变量等数据,若空间不足会抛出java.lang.OutOfMemoryError: PermGen space错误。 在Java 8之后,永久代已被元空间取代,元空间直接使用本地操作系统内存,可能出现java.lang.OutOfMemoryError: Metaspace。 解决方案:增大永久代或元空间的大小,检查代码是否有大量动态加载类或者反射操作生成过多类信息的情况。


栈空间溢出: 错误信息: java.lang.StackOverflowError 原因:递归调用过深或线程栈帧过大,导致线程栈空间耗尽。 解决方案:调整栈的大小(-Xss参数),改进算法避免深度递归,合理控制线程数量或每个线程栈的大小。


首先栈溢出定位很简单,直接异常栈就会告诉,去调整代码逻辑即可。这里着重要聊一下的是元空间溢出和堆溢出。


元空间溢出


JDK1.8及其以后版本,元空间替代了永久代,其主要用于存储类的元数据信息,包括类的结构信息(如字段、方法、接口、常量池等)、运行时常量池、方法字节码、静态变量等。也就是说类被加载了,其相关信息就会存在元空间中。


此处也许有些读者会有疑问:


类是要在被用到的时候才会加载,也就是一般我们new对象的时候对应的类才会被加载,那么存在元空间在堆之前被撑爆的情况吗?


答:


当然是存在的,只要你的元空间比你的堆小,或者频繁用Class.forName()、ClassLoader.loadClass()等反射的手法来加载类,但是不new对象,也能把元空间撑爆了。


元空间溢出其实是比较难遇见的,但是定位方法其实不难,直接代码全局搜Class.forName之类的语法基本就能定位元凶。


接下来本文要讲的重点是生产中最容易遇见的一种JVM内存溢出——堆溢出,以及比堆溢出藏得更深的隐形杀手——内存泄漏,这两者如何定位以及解决。

2.堆溢出、内存泄定位及解决办法

2.1.示例代码

直接的堆溢出从异常栈信息里是能看出哪里造成的OOM,很容易定位:

难定位的是哪种?怕的是内存泄漏,也就是堆还没有撑爆,但是就是在要爆不爆之间徘徊,造成频繁的GC,由于GC的时候是要”Stop The World“,会暂停所有JAVA线程的工作,这自然会浪费CPU资源,外界的感知就是”变慢了“。这里我们详细的来聊一聊如何定位内存泄漏的问题。


测试代码:

//定义一个类,该类中一旦调用一个方法,就会持续让List持有一个个1024KB大小的内存空间,但是又不会直接撑爆heap
import java.lang.management.ManagementFactory;
import java.lang.management.MemoryMXBean;
import java.util.ArrayList;
public class MyUser {
    private Byte[] bytes;
    ArrayList<byte[]> list = new ArrayList();
    public static final int OBJECT_SIZE = 1024 * 1024; // 每个对象占用1MB空间
    public static final double HEAP_UTILIZATION_RATIO = 0.8; // 目标堆利用率
    public MyUser(){}
    public void callTest(){
        try {
            MemoryMXBean memoryMxBean = ManagementFactory.getMemoryMXBean();
            long maxMemory = Runtime.getRuntime().maxMemory();
            while (true) {
                // 检查当前堆内存使用情况,如果已超过目标利用率,则退出循环
                long currentHeapUsage = memoryMxBean.getHeapMemoryUsage().getUsed();
                if ((double) currentHeapUsage / maxMemory >= HEAP_UTILIZATION_RATIO) {
                    System.out.println("当前堆内存使用率达到目标利用率,程序即将退出...");
                    break;
                }
                // 创建一个大对象
                byte[] largeObject = new byte[OBJECT_SIZE];
                // 将大对象添加到列表中
                list.add(largeObject);
                // 添加一个延时,模拟其他操作,便于观察
                Thread.sleep(100); // 等待100毫秒
                // 可以在此处添加额外的日志输出或监控代码,用于记录GC信息和内存状态
            }
            // 清理资源,防止后续分析时误判
            list.clear();
        } catch (InterruptedException e) {
            throw new RuntimeException(e);
        }
    }
}
//用junit开始测试
@Test
public void test02() throws Exception{
  MyUser myUser = new MyUser();
  myUser.callTest();

2.2.抓堆快照

运行上面的示例代码,打开jvisualVM,开始监控程序,可以看到:

垃圾回收是有在进行的,而且频率不低,但是堆的大小一直在扩,被占用率一直在攀升,说明并没有被回收掉,存在严重的内存泄露。

这时候就需要把堆的dump抓下来看看了。右上角有抓heap dump的选项。

2.3.分析堆快照

通过MAT看看了,看看到底是有哪些东西占着内存一直没被回收掉。

MAT是eclipse旗下的一款heap的分析工具,可以用来专门分析heap dump。下载地址:

Eclipse Memory Analyzer | projects.eclipse.org

MAT和JDK有版本适配关系!千万别下错了,作者用的JDK,下载的1.8.1版本:

还要注意下载的安装包和操作系统之间也是有严格的适配关系的,作者第一次就下成了x86而不是x86_64,确定好自己的操作系统平台下对应的:

用MAT打开抓下来的heap dump:

工具会分析显示除怀疑内存泄漏的地方:

其实从饼状图上已经可以看出端倪,饼状图显示了可能存在内存泄漏的对象和该对象持有的内存的对比,这个对象自身占的内存大小只有浅灰色的一小条,但是其持有的内存居然达到了300多MB,很明显的内存泄漏的情况。

我们初步断定了存在内存泄漏,接下来当然是要定位具体位置,然后才好解决它。

展开详细信息,可以看到怀疑是Main线程上的一个MyUser类型里面的一个类型为ArrayList名叫list的成员变量造成了内存泄漏:

接下来去看这里的代码逻辑就行了。

目录
相关文章
|
9月前
|
Arthas 存储 算法
深入理解JVM,包含字节码文件,内存结构,垃圾回收,类的声明周期,类加载器
JVM全称是Java Virtual Machine-Java虚拟机JVM作用:本质上是一个运行在计算机上的程序,职责是运行Java字节码文件,编译为机器码交由计算机运行类的生命周期概述:类的生命周期描述了一个类加载,使用,卸载的整个过类的生命周期阶段:类的声明周期主要分为五个阶段:加载->连接->初始化->使用->卸载,其中连接中分为三个小阶段验证->准备->解析类加载器的定义:JVM提供类加载器给Java程序去获取类和接口字节码数据类加载器的作用:类加载器接受字节码文件。
822 55
|
4月前
|
存储 缓存 Java
我们来说一说 JVM 的内存模型
我是小假 期待与你的下一次相遇 ~
373 5
|
4月前
|
存储 缓存 算法
深入理解JVM《JVM内存区域详解 - 世界的基石》
Java代码从编译到执行需经javac编译为.class字节码,再由JVM加载运行。JVM内存分为线程私有(程序计数器、虚拟机栈、本地方法栈)和线程共享(堆、方法区)区域,其中堆是GC主战场,方法区在JDK 8+演变为使用本地内存的元空间,直接内存则用于提升NIO性能,但可能引发OOM。
|
10月前
|
Arthas 监控 Java
Arthas memory(查看 JVM 内存信息)
Arthas memory(查看 JVM 内存信息)
810 6
|
10月前
|
监控 Java Unix
6个Java 工具,轻松分析定位 JVM 问题 !
本文介绍了如何使用 JDK 自带工具查看和分析 JVM 的运行情况。通过编写一段测试代码(启动 10 个死循环线程,分配大量内存),结合常用工具如 `jps`、`jinfo`、`jstat`、`jstack`、`jvisualvm` 和 `jcmd` 等,详细展示了 JVM 参数配置、内存使用、线程状态及 GC 情况的监控方法。同时指出了一些常见问题,例如参数设置错误导致的内存异常,并通过实例说明了如何排查和解决。最后附上了官方文档链接,方便进一步学习。
1516 4
|
存储 设计模式 监控
快速定位并优化CPU 与 JVM 内存性能瓶颈
本文介绍了 Java 应用常见的 CPU & JVM 内存热点原因及优化思路。
1178 166
|
11月前
|
存储 缓存 算法
JVM简介—1.Java内存区域
本文详细介绍了Java虚拟机运行时数据区的各个方面,包括其定义、类型(如程序计数器、Java虚拟机栈、本地方法栈、Java堆、方法区和直接内存)及其作用。文中还探讨了各版本内存区域的变化、直接内存的使用、从线程角度分析Java内存区域、堆与栈的区别、对象创建步骤、对象内存布局及访问定位,并通过实例说明了常见内存溢出问题的原因和表现形式。这些内容帮助开发者深入理解Java内存管理机制,优化应用程序性能并解决潜在的内存问题。
543 29
JVM简介—1.Java内存区域
|
11月前
|
缓存 监控 算法
JVM简介—2.垃圾回收器和内存分配策略
本文介绍了Java垃圾回收机制的多个方面,包括垃圾回收概述、对象存活判断、引用类型介绍、垃圾收集算法、垃圾收集器设计、具体垃圾回收器详情、Stop The World现象、内存分配与回收策略、新生代配置演示、内存泄漏和溢出问题以及JDK提供的相关工具。
JVM简介—2.垃圾回收器和内存分配策略
|
11月前
|
存储 设计模式 监控
如何快速定位并优化CPU 与 JVM 内存性能瓶颈?
如何快速定位并优化CPU 与 JVM 内存性能瓶颈?
315 0
如何快速定位并优化CPU 与 JVM 内存性能瓶颈?
|
12月前
|
存储 算法 Java
JVM: 内存、类与垃圾
分代收集算法将内存分为新生代和老年代,分别使用不同的垃圾回收算法。新生代对象使用复制算法,老年代对象使用标记-清除或标记-整理算法。
186 6