解决 Java 内存泄露处理细节

现象

最近项目组从NET平台迁移到Java的Dubbo平台上,由于大家都是Java的生手,发生了蛮多的问题,以后一一记录。现在解决一个遇到的关于Java程序内存泄露的问题。

版本

  • Java 1.8
  • Dubbo 2.6.2
  • Docker 18.0.2

系统环境

我们这里是Docker Swarm集群,三台机器组成,Dubbo服务随机部署到三台机器上。

问题重现

上线了一个Dubbo服务,这个服务涉及到数据库查询、排序分析、第三方接口调用。 服务启动初始内存占用500MB左右,每检索一次,内存增加10MB到几十MB不等,而且不释放。持续增高,最高可以塞满整个服务器的内存。

检查问题

首先,由于我们是部署在Docker集群上的,所以得去容器内进行检查,刚上线,所以基础容器选择的是JDK版本,没有用JRE。因为JDK带有很多的调试工具。

查看生产环境并导出heap.hprof

首先看看容器的运行情况:

docker stats
复制代码

这是后面调试的时候截的图,早期发现的时候,内存是4.8G。明显内存占用过多了。

因为宿主是没有任何java环境的,进容器内做内存分析。

docker exec -it 容器ID bash # 进入指定容器
jmap -histo 1 | head -n 30 # 通过jmap工具查看
复制代码

jmap的检查结果当时忘了截图了,这里就不留存了,百度能搜到了,就一笔带过。

现在导出heap.hprof文件,方便用MAT进行分析。

jmap -dump:format=b,file=heap.hprof 1
复制代码

在docker容器内,PID 1 是服务进程,以上命令将会在当前目录生成heap.hprof文件,比较大,我的有1.2G。可以先压缩了,再传回来,进行分析。

使用MAT进行内存分析

独立版下载地址:MAT

打开从服务器下载回来的heap.hprof文件

点击Leak Suspects查看分析结果。

点击Details查看详情

上图可以看到,MAT分析结果表明,OcMapperFactory这个类有问题。

看看具体代码:

这个OcMapperFactory是用来封装orika的工具类,而orika是一个对象映射工具。由于这里没有用单例导致了内存的泄露。加上单例再看看:

重新测试,发现内存已经稳定。

转载于:https://juejin.im/post/5b63ba9f6fb9a04f9e232e50

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值