Chan Chen Coding...

          Eclipse Memory Analyzer介紹

          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;

          posted on 2012-08-30 13:58 Chan Chen 閱讀(3859) 評論(0)  編輯  收藏 所屬分類: Scala / Java

          主站蜘蛛池模板: 抚远县| 延安市| 简阳市| 武川县| 榆社县| 仙桃市| 香港| 颍上县| 建始县| 凤冈县| 富源县| 平利县| 曲靖市| 乐至县| 洛宁县| 崇明县| 信阳市| 泰顺县| 栾城县| 西藏| 禹州市| 大港区| 晴隆县| 抚远县| 洛川县| 舒城县| 华蓥市| 明水县| 延长县| 阜宁县| 贺州市| 凉山| 淳安县| 焦作市| 通江县| 金川县| 彭阳县| 长白| 江孜县| 武邑县| 手游|