JVM內存回收問題處理方法
本文和大家重點討論一下JVM內存回收問題的解決方法,通常我們說的JVM內存回收總是在指堆內存回收,確實只有堆中的內容是動態(tài)申請分配的,所以以上對象的年輕代和年老代都是指的JVM的Heap空間,而持久代則是之前提到的MethodArea,不屬于Heap。
JVM內存回收
了解JVM的系統(tǒng)結構,再來看看JVM內存回收問題了——
Sun的JVMGenerationalCollecting(垃圾回收)原理是這樣的:把對象分為年青代(Young)、年老代(Tenured)、持久代(Perm),對不同生命周期的對象使用不同的算法。(基于對對象生命周期分析)
如上圖所示,為Java堆中的各代分布。
1.Young(年輕代)
年輕代分三個區(qū)。一個Eden區(qū),兩個Survivor區(qū)。大部分對象在Eden區(qū)中生成。當Eden區(qū)滿時,還存活的對象將被復制到Survivor區(qū)(兩個中的一個),當這個Survivor區(qū)滿時,此區(qū)的存活對象將被復制到另外一個Survivor區(qū),當這個Survivor去也滿了的時候,從***個Survivor區(qū)復制過來的并且此時還存活的對象,將被復制年老區(qū)(Tenured。需要注意,Survivor的兩個區(qū)是對稱的,沒先后關系,所以同一個區(qū)中可能同時存在從Eden復制過來對象,和從前一個Survivor復制過來的對象,而復制到年老區(qū)的只有從***個Survivor去過來的對象。而且,Survivor區(qū)總有一個是空的。
2.Tenured(年老代)
年老代存放從年輕代存活的對象。一般來說年老代存放的都是生命期較長的對象。
3.Perm(持久代)
用于存放靜態(tài)文件,如今Java類、方法等。持久代對垃圾回收沒有顯著影響,但是有些應用可能動態(tài)生成或者調用一些class,例如Hibernate等,在這種時候需要設置一個比較大的持久代空間來存放這些運行過程中新增的類。持久代大小通過-XX:MaxPermSize=進行設置。
舉個例子:當在程序中生成對象時,正常對象會在年輕代中分配空間,如果是過大的對象也可能會直接在年老代生成(據(jù)觀測在運行某程序時候每次會生成一個十兆的空間用收發(fā)消息,這部分內存就會直接在年老代分配)。年輕代在空間被分配完的時候就會發(fā)起內存回收,大部分內存會被回收,一部分幸存的內存會被拷貝至Survivor的from區(qū),經(jīng)過多次回收以后如果from區(qū)內存也分配完畢,就會也發(fā)生內存回收然后將剩余的對象拷貝至to區(qū)。等到to區(qū)也滿的時候,就會再次發(fā)生內存回收然后把幸存的對象拷貝至年老區(qū)。
通常我們說的JVM內存回收總是在指堆內存回收,確實只有堆中的內容是動態(tài)申請分配的,所以以上對象的年輕代和年老代都是指的JVM的Heap空間,而持久代則是之前提到的MethodArea,不屬于Heap。
關于JVM內存管理的一些建議
1、手動將生成的無用對象,中間對象置為null,加快內存回收。
2、對象池技術如果生成的對象是可重用的對象,只是其中的屬性不同時,可以考慮采用對象池來較少對象的生成。如果有空閑的對象就從對象池中取出使用,沒有再生成新的對象,大大提高了對象的復用率。
3、JVM調優(yōu)通過配置JVM的參數(shù)來提高垃圾回收的速度,如果在沒有出現(xiàn)內存泄露且上面兩種辦法都不能保證JVM內存回收時,可以考慮采用JVM調優(yōu)的方式來解決,不過一定要經(jīng)過實體機的長期測試,因為不同的參數(shù)可能引起不同的效果。如-Xnoclassgc參數(shù)等。
推薦的兩款JVM內存檢測工具
1、jconsoleJDK自帶的內存監(jiān)測工具,路徑jdkbin目錄下jconsole.exe,雙擊可運行。連接方式有兩種,***種是本地方式如調試時運行的進程可以直接連,第二種是遠程方式,可以連接以服務形式啟動的進程。遠程連接方式是:在目標進程的jvm啟動參數(shù)中添加-Dcom.sun.management.jmxremote.port=1090-Dcom.sun.management.jmxremote.ssl=false-Dcom.sun.management.jmxremote.authenticate=false1090是監(jiān)聽的端口號具體使用時要進行修改,然后使用IP加端口號連接即可。通過該工具可以監(jiān)測到當時內存的大小,CPU的使用量以及類的加載,還提供了手動gc的功能。優(yōu)點是效率高,速度快,在不影響進行運行的情況下監(jiān)測產品的運行。缺點是無法看到類或者對象之類的具體信息。使用方式很簡單點擊幾下就可以知道功能如何了,確實有不明白之處可以上網(wǎng)查詢文檔。
2、JProfiler收費的工具,但是到處都有破解辦法。安裝好以后按照配置調試的方式配置好一個本地的session即可運行。可以監(jiān)測當時的內存、CPU、線程等,能具體的列出內存的占用情況,還可以就某個類進行分析。優(yōu)點很多,缺點太影響速度,而且有的類可能無法被織入方法,例如我使用jprofiler時一直沒有備份成功過,總會有一些類的錯誤。
【編輯推薦】
- 探索Java工作原理之JVM內存回收
- Tomcat配置JVM參數(shù)巧妙方法
- JVM基礎:解析JVM分代垃圾回收策略
- JVM for Linux JIT診斷技術簡介
- 巧解IBM JVM for Linux onPOWER性能調優(yōu)