Java内存溢出问题排查分析

目录


前言

项目运行过程中,我们可能会遇到Java内存溢出Out Of Memory。此时我们可以借助内存分析工具MAT(Memory Analyzer Tool),来定位是哪里出现了问题。


一、MAT(Memory Analyzer Tool

下载地址:Eclipse Memory Analyzer Open Source Project | The Eclipse Foundation

注意:JDK-8建议使用11版本,否则会提示版本不兼容

二、软件初识

解压后目录内有个MemoryAnalyzer.ini文件,该文件里面有个-Xmx参数。该参数表示最大内存占用量,默认为1024m。
建议修改为小于本机内存大小,大于要分析的dump文件大小
-startup
plugins/org.eclipse.equinox.launcher_1.5.0.v20180512-1130.jar
--launcher.library
plugins/org.eclipse.equinox.launcher.win32.win32.x86_64_1.1.700.v20180518-1200
-vmargs
### -Xmx1024my原本默认为1024m,此处我修改为4096m
-Xmx4096m

三、捕获dump文件

        首先在程序中模拟出一段内存溢出的逻辑:

    @RequestMapping("/testOutOfMemory")
    @ResponseBody
    public void testOutOfMemory() throws Exception {
        String name = "Aikes";
        for (int i = 0; i < 10000000; i++) {
            name += name;
        }
        System.out.println(name);
    }

        然后启动项目,开始准备捕获dump文件。这里的捕获方式分为两种,一种是主动捕获,一种是被动捕获:

1、主动方式

        顾名思义,当内存溢出发生后,通过指令的方式手机当前应用程序下的内存使用情况。

1、通过(Linux) ps -ef|grep find 或者 (Dos)netstat -ano|findstr 查找java程序运行的PID

2、使用指令收集dump:jmap -dump:format=b,file=路径/heapdump.hprof  查到的PID

注意:主动获取dump文件必须是一出现内存异常就获取dump文件,这样获取的文件信息才比较准确。如果无法及时获取,推荐通过第二种方式获取dump文件。

2、被动方式

        该方式是启动Java服务时,增加额外参数。当程序发生内存溢出时自动收集dump文件:

1、-XX:+HeapDumpOnOutOfMemoryError -XX:HeapDumpPath=/opt/oom/heapdump.hprof

        配置好后,调用我们预先模拟的内存溢出接口

        稍等片刻控制台开始打印日志,提示出现内存溢出并且已经收集到dump文件到指定目录:

四、分析dump文件

        打开下载好的mat软件,通过file-open打开抓取到的dump文件(hprof文件)

        点击切换视图,可以看到内存占用百分之八十是因为这个线程,继续点开发现是一个超大的字符串"AikesAikesAikes...."

        此时我们已经发现了内存溢出的直接原因,接下来要寻找出现这个问题的代码在哪里。再返回到最初的大饼图,点击最下面的details。然后点击See stacktrace 堆叠追踪。

        这里可以看到完整的堆栈信息,里面可以发现我们增加模拟溢出代码的那个Java文件,并且爆发内存溢出的代码行也可以对上,至此溢出分析结束。


总结

        模拟的内存溢出针对性很强,并且我们抓取dump文件也很及时,所以在分析的时候很简单。实际使用过程中面对的陷阱很多,需要从诸多可能中排查幕后凶手。Mat工具功能还很多,目前只是粗略的使用,后续如果有新的发现会继续补充到博文中。

  • 9
    点赞
  • 71
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论
Java内存溢出指的是在Java程序运行过程中,因为申请的内存超出了可用内存的限制,导致程序终止的现象。下面是一些排查Java内存溢出问题方法: 1.确定是否是内存溢出问题:查看程序的错误日志或异常信息,如果有OutOfMemoryError的错误信息,可以确定是内存溢出问题。 2.分析问题发生的位置:追踪错误日志或异常堆栈,定位到代码中可能导致内存溢出的地方,比如不断创建对象、大量递归调用等。 3.查看程序的内存使用情况:可以使用Java的内存分析工具,如jmap、jstat等,查看程序运行时的内存使用情况,包括堆内存和非堆内存的使用情况。 4.检查代码中是否有资源未释放:Java中需要手动释放的资源包括文件流、数据库连接等,如果资源没有正确释放,会导致内存泄漏,最终导致内存溢出。 5.检查是否存在循环引用:循环引用指的是多个对象相互引用,导致垃圾回收器无法回收它们,最终导致内存溢出。可以使用内存分析工具来分析程序中是否存在循环引用的情况。 6.调整JVM参数:可以通过调整JVM的参数来增加可用内存,比如增加堆内存的大小。可以使用命令行参数'-Xms'和'-Xmx'来指定初始堆大小和最大堆大小。 7.优化代码:检查代码中是否存在不必要的对象创建、频繁的垃圾回收等问题,优化程序的设计和算法,减少内存使用。 8.升级JDK版本:某些JDK版本中可能存在内存泄漏或其他内存相关的问题,升级到最新的JDK版本可以解决一些内存溢出问题。 总之,排查Java内存溢出问题需要分析错误日志、查看内存使用情况、检查代码和资源释放等等,找出问题的根源并及时修复。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Aikes902

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值