Eclipse Memory Analyzer是一個非常棒的堆內存分析工具,是JDK自帶的堆分析工具jhat的一個非常好的替代品,能夠快速地定位Java內存泄露的原因。
可能有的同學會問,JVM不是號稱自動內存管理,GC會自動垃圾回收,Java怎么會有內存泄露,不會搞錯吧?當然不會^_^, Java的內存泄露不同于C/C++的內存泄露,C/C++的內存泄露是由于使用了堆內存(new/malloc)卻沒有釋放(delete/free),導致無法再使用到該內存片,而Java的內存泄露是無謂地引用了一些垃圾的對象,譬如我們有一個Map對象,不斷往里面放對象,實際的場景可能是這些對象不會再被使用到,這時候,這部分數據本身是垃圾的(因為不會再被使用),但實際上JVM會不會釋放它(因為還被Map)引用著,這就是Java的內存泄露。
在開始分析之前,我們先想想,在編程這個角度上,我們如何避免堆內存泄露呢?實際上java.lang.ref這個包已經為我們提供了一種問題解決方案。Java的引用有4種:強引用(Strong Reference)、軟引用(Soft Reference)、弱引用(Weak Reference)、幻影引用(Phantom Reference),關于這部分介紹的文章一大批,此處就不做深入,我們只需要知道如下的信息:
- 強引用:除非將引用置為null,否則JVM不會對它垃圾,這是最常用的引用方式
- 軟引用:在堆內存不足的時候,GC會將其垃圾回收
- 弱引用:每次GC都會將其垃圾回收
- 幻影引用:跟沒有引用一樣,每次獲得的都是空的,沒有太多使用的意義,僅是為了追蹤對象在JVM的狀態
一般對于大數據量的Cache信息或大對象,使用軟引用/弱引用是一種非常好的習慣,或者至少使用一種淘汰算法,避免在堆內存擁擠大量的對象導致內存不足,如下是兩個非常好的JDK默認提供的HashMap替代者:
- org.apache.commons.collections.map.ReferenceMap:支持強引用/軟引用和弱引用來存儲key/value對
- org.apache.commons.collections.map.LRUMap:可以控制總容量,采用LRU淘汰算法,將不常使用的數據淘汰出去
介紹完一些背景,我們開始進入主題。在開始分析之前,我們需要先dump下JVM的堆內存信息(雖然Eclipse Memory Analyzer直接attach到JVM上獲取棧再分析,實際應用價值不大)
jmap –dump:file=test.bin {pid}
現在我們有了test.bin這個堆文件,使用Eclipse Memory Analyzer打開,分析完堆,我們可以選擇“Leak Suspects Report”進行內存泄露分析。通過這個視圖,我們可以大概得到內存泄露的初步結論

Historygram也是一個非常常用的視圖,可以獲得堆中對象的數據統計,有排序、過濾的功能,非常好用

Eclipse Memory Analyzer還包括如下功能:
- 在Historygram視圖中右擊對象彈出的功能框中,可以獲得對象相互引用的關系的功能
- Dominator Tree的視圖采用Tree的方式來展現整個棧對象相互引用的情況
- OQL視圖支持使用OQL語言來查詢對象信息
-----------------------------------------------------
Silence, the way to avoid many problems;
Smile, the way to solve many problems;