如何诊断 Java 中的内存泄露
2023-03-09 21:57:09 时间
每次我怀疑有内存泄漏时,我都要翻箱倒柜找这些命令。所以,这里总结一下以备后用:
首先,我用下面的命令监视进程:
- while ( sleep 1 ) ; do ps -p $PID -o %cpu,%mem,rss ; done
(如果有的话还有New Relic)
如果你看到内存上升很快,可能是因为虚拟机设置。如果你没有明确指定JVM的内存设置,它将设置默认值给他们。要获得默认值,使用以下命令:
- java -XX:+PrintFlagsFinal -version | grep -i HeapSize
如果这些都不符合你所希望的,那么你就需要指定JVM的内存设置。可以用下面的命令设置最小和***堆大小:
- java -Xms128m -Xmx256m
尽管你有了合理的内存设置,也可以监控进程,但你仍然可能看到内存随时间增加。为了进一步探究原因,你可以使用下面的命令查看对象实例的直方图:
jmap -histo $PID
如果仍然没有足够的信息,那么可以用以下命令进行堆转储:
jmap -dump:format=b,file=/tmp/dump1.hprof $PID
通常,我会用两个堆转储,然后使用下面的jhat命令比较它们:
jhat -baseline /tmp/dump1.hprof /tmp/dump2.hprof
这个命令会启动一个HTTP服务器,你可以用它来探索这两个堆转储之间的差值。在默认情况下,HTTP服务器启动7000端口,你可以在浏览器中访问该端口。
如果你有防火墙,可以通过SSH访问,那么你可以通过如下命令连接该端口:
ssh -L 7000:localhost:7000 $HOST
向下滚动到***页的底部,你会看到两个有用的链接:
这将给你展示在不同堆转储之间所有“新”的实例,应该对你检测泄漏来自哪里有些帮助。截图如下:
然后你就拥有了一个神奇命令行的快速查看目录,以便于你需要诊断内存泄漏时使用(然而我总是忘记)。
相关文章
- Java要抛弃祖宗的基业,Java程序员危险了!
- 十大 Java 语言特性
- JVM 三色标记算法,原来是这么回事!
- 聊聊 Spring 事务控制策略以及 @Transactional 失效问题避坑
- 写给 Java 程序员的前端 Promise 教程
- 写给 Java 程序员的前端 Promise 教程,你学会了吗?
- Java 中为什么不全部使用 Static 方法?
- Java 池化技术你了解多少?
- Java 服务 Docker 容器化优秀实践
- Spring Boot + EasyExcel导入导出,简直太好用了!
- 我们一起聊聊 Java 内存泄漏
- CentOS 下安装 Docker 极简教程
- JDK 19 功能集冻结:Java 19 只有七个新特性
- 关于 CMS 垃圾回收器,你真的懂了吗?
- 为什么会有这么多编程语言?
- 改善Java代码的八个建议
- 接口流量突增,如何做好性能优化?
- Java 以编程方式创建JAR文件
- POJO、Java Bean是如何定义的
- Spring 的 Bean 明明设置了 Scope 为 Prototype,为什么还是只能获取到单例对象?