0%
这是一片思考的空间 -- arthinking
Spring 重构&代码整洁之道 软件设计 JVM 并发编程 数据结构与算法 分布式 存储 网络 微服务 设计模式
Java技术栈 - 涉及Java技术体系

Java中内存泄露的分析

小哥最近在分析项目中的内存泄露情况,使用了Eclipse Memory Analyzer,移步到这里下载最新版本:

http://www.eclipse.org/mat/

我们知道,不同版本的虚拟机和JDK版本,垃圾回收机制和内存空间布局都是有所差别的,我使用的是这个版本的JDK

java version "1.6.0_26"

Java(TM) SE Runtime Environment (build 1.6.0_26-b03)

Java HotSpot(TM) Client VM (build 20.1-b02, mixed mode, sharing)

1、下面监控一下我开发环境下的Resin服务器:

首先,运行JDK bin目录下的jconsole.exe文件,链接到Resin服务器:

20141110001

进入之后,就可以查看到堆内存,类,CPU的使用情况了。

细心的读者发现了我使用的是主流的HotSpot虚拟机。选择 com.sun.management-->HotSpotDiagnostic-->操作-->dumpHeap(堆转储),在操作调用中的p0填写导出目录,点击dumpHeap即可:

20141110002

这样就可以在p0对应的输出目录中找打导出的文件了:dumpHeap.hprof

接下来启动MemoryAnalyzer, File --> Open File... 选择刚才导出的文件进行分析。

选择目录中的Path To GC Roots --> exclude weak/soft references 排除掉弱引用,剩下的才是最有可能导致内存泄露的问题:

20141110003

目前查看到的都是classloader加载到的Class文件的引用,并没有发现异常信息:

20141110004

继续观察,发现问题随时分析。

哦对了,ClassLoader也是有可能产能内存泄露的哦,详细请查看我的这篇文章:

How to creating a memory leak with Java

2、监控一段测试程序:

可以通过配置Java的 -XX:+HeapDumpOnOutOfMemoryError 参数让虚拟机在出现内存溢出异常时Dump出当前的内存堆转储快照。

下面是来自深入理解Java虚拟机的例子:

/**

  • VM Args:-verbose:gc -Xms20m -Xmx20m -XX:+HeapDumpOnOutOfMemoryError
  • 限制Java堆大小为20M,不可扩展,让虚拟机在出现内存溢出异常时Dump出当前的内存堆转储快照
  • @Author zzm
    */
    public class HeapOOM{
    static class OOMObject{
    }
    public static void main(String[] args){
    List list = new ArrayList();
    while(true){
    list.add(new OOMObject());
    }
    }
    }

执行后可以发现转储快照被导出了:

Dumping heap to java_pid4148.hprof ...

Heap dump file created [24728647 bytes in 0.342 secs]

Exception in thread "main" java.lang.OutOfMemoryError: Java heap space

at java.util.Arrays.copyOf(Unknown Source)

at java.util.Arrays.copyOf(Unknown Source)

at java.util.ArrayList.ensureCapacity(Unknown Source)

at java.util.ArrayList.add(Unknown Source)

at me.arthinking.memoryleaktest.HeapOOM.main(HeapOOM.java:18)

接下来使用MemoryAnalyzer分析这个文件就可以了。

如果是内存泄露,可以进一步通过查看泄露对象到GC Roots的引用链。

如果不存在内存泄露,就应当坚持虚拟机的堆参数(-Xmx和-Xms)。

欢迎关注我的其它发布渠道

订阅IT宅
内功修炼
Java技术栈
Java架构杂谈是IT宅精品文章公众号,欢迎订阅:
📄 网络基础知识:两万字长文50+张趣图带你领悟网络编程的内功心法 📄 HTTP发展史:三万长文50+趣图带你领悟web编程的内功心法 📄 HTTP/1.1:可扩展,可靠性,请求应答,无状态,明文传输 📄 HTTP/1.1报文详解:Method,URI,URL,消息头,消息体,状态行 📄 HTTP常用请求头大揭秘 📄 HTTPS:网络安全攻坚战 📄 HTTP/2:网络安全传输的快车道 📄 HTTP/3:让传输效率再一次起飞 📄 高性能网络编程:图解Socket核心内幕以及五大IO模型 📄 高性能网络编程:三分钟短文快速了解信号驱动式IO 📄 高性能网络编程:彻底弄懂IO复用 - IO处理杀手锏,带您深入了解select,poll,epoll 📄 高性能网络编程:异步IO:新时代的IO处理利器 📄 高性能网络编程:网络编程范式 - 高性能服务器就这么回事 📄 高性能网络编程:性能追击 - 万字长文30+图揭秘8大主流服务器程序线程模型
📄 Java内存模型:如果有人给你撕逼Java内存模型,就把这些问题甩给他 📄 一文带你彻底理解同步和锁的本质(干货) 📄 AQS与并发包中锁的通用实现 📄 ReentrantLock介绍与使用 📄 ReentrantReadWriteLock介绍与使用 📄 ReentrantLock的Condition原理解析 📄 如何优雅的中断线程 📄 如何优雅的挂起线程 📄 图解几个好玩的并发辅助工具类 📄 图解BlockingQueue阻塞队列
📄 消息队列那么多,为什么建议深入了解下RabbitMQ? 📄 高并发异步解耦利器:RocketMQ究竟强在哪里? 📄 Kafka必知必会18问:30+图带您看透Kafka
📄 洞悉MySQL底层架构:游走在缓冲与磁盘之间 📄 SQL运行内幕:从执行原理看调优的本质 📄 洞悉Redis技术内幕:缓存,数据结构,并发,集群与算法