JVM分代垃圾回收策略的基礎概念

1、爲何要分代算法

     分代的垃圾回收策略,是基於這樣一個事實:不一樣的對象的生命週期是不同的。所以,不一樣生命週期的對象能夠採起不一樣的收集方式,以便提升回收效率。spa

     在Java程序運行的過程當中,會產生大量的對象,其中有些對象是與業務信息相關,好比Http請求中的Session對象、線程、Socket鏈接,這類對象跟業務直接掛鉤,所以生命週期比較長。可是還有一些對象,主要是程序運行過程當中生成的臨時變量,這些對象生命週期會比較短,好比:String對象,因爲其不變類的特性,系統會產生大量的這些對象,有些對象甚至只用一次便可回收。線程

     試想,在不進行對象存活時間區分的狀況下,每次垃圾回收都是對整個堆空間進行回收,花費時間相對會長,同時,由於每次回收都須要遍歷全部存活對象,但實際上,對於生命週期長的對象而言,這種遍歷是沒有效果的,由於可能進行了不少次遍歷,可是他們依舊存在。所以,分代垃圾回收採用分治的思想,進行代的劃分,把不一樣生命週期的對象放在不一樣代上,不一樣代上採用最適合它的垃圾回收方式進行回收。3d

2、如何分代對象

如圖所示:blog

     虛擬機中的共劃分爲三個代:年輕代(Young Generation)、年老代(Old Generation)和持久代(Permanent Generation)。其中持久代主要存放的是Java類的類信息,與垃圾收集要收集的Java對象關係不大。年輕代和年老代的劃分是對垃圾收集影響比較大的。生命週期

年輕代:內存

     全部新生成的對象首先都是放在年輕代的。年輕代的目標就是儘量快速的收集掉那些生命週期短的對象。年輕代分三個區。一個Eden區,兩個Survivor區(通常而言)。大部分對象在Eden區中生成。當Eden區滿時,還存活的對象將被複制到Survivor區(兩個中的一個),當這個Survivor區滿時,此區的存活對象將被複制到另一個Survivor區,當這個Survivor區也滿了的時候,從第一個Survivor區複製過來的而且此時還存活的對象,將被複制「年老區(Tenured)」。須要注意,Survivor的兩個區是對稱的,沒前後關係,因此同一個區中可能同時存在從Eden複製過來的對象,和從前一個Survivor複製過來的對象,而複製到年老區的只有從第一個Survivor區過來的對象。並且,Survivor區總有一個是空的。同時,根據程序須要,Survivor區是能夠配置爲多個的(多於兩個),這樣能夠增長對象在年輕代中的存在時間,減小被放到年老代的可能。虛擬機

     新生代有劃分爲Eden、From Survivor和To Survivor三個部分,他們對應的內存空間的大小比例爲8:1:1,也就是,爲對象分配內存的時候,首先使用Eden空間,通過GC後,沒有被回收的會首先進入From Survivor區域,任什麼時候候,都會保持一個Survivorq區域(From Survivor或To Survivor)徹底空閒,也就是說新生代的內存利用率最大爲90%。From Survivor和To Survivor兩個區域會根據GC的實際狀況,進行互換,將From Survivor區域中的對象所有複製到To Survivor區域中,或者反過來,將To Survivor區域中的對象所有複製到From Survivor區域中。io

年老代:

     在年輕代中經歷了N次垃圾回收後仍然存活的對象,就會被放到年老代中。所以,能夠認爲年老代中存放的都是一些生命週期較長的對象。

     GC過程當中,當某些對象通過屢次GC都沒有被回收,可能會進入到年老代。或者,當新生代沒有足夠的空間來爲對象分配內存時,可能會直接在年老代進行分配。

持久代:

     用於存放靜態文件,現在Java類、方法等。持久代對垃圾回收沒有顯著影響,可是有些應用可能動態生成或者調用一些class,例如Hibernate等,在這種時候須要設置一個比較大的持久代空間來存放這些運行過程當中新增的類。持久代大小經過-XX:MaxPermSize=<N>進行設置。

    永久代實際上對應着虛擬機運行時數據區的「方法區」,這裏主要存放類信息、靜態變量、常量等數據。通常狀況下,永久代中對應的對象的GC效率很是低,由於這裏的的大部分對象在運行都不要進行GC,它們會一直被利用,直到JVM退出。

3、什麼狀況下觸發垃圾回收

因爲對象進行了分代處理,所以垃圾回收區域、時間也不同。GC有兩種類型:Scavenge GC和Full GC

Scavenge GC

    通常狀況下,當新對象生成,而且在Eden申請空間失敗時,就會觸發Scavenge GC,對Eden區域進行GC,清除非存活對象,而且把尚且存活的對象移動到Survivor區。而後整理Survivor的兩個區。這種方式的GC是對年輕代的Eden區進行,不會影響到年老代。由於大部分對象都是從Eden區開始的,同時Eden區不會分配的很大,因此Eden區的GC會頻繁進行。於是,通常在這裏須要使用速度快、效率高的算法,使Eden去能儘快空閒出來。

     對整個堆進行整理,包括Young、Tenured和Perm。Full GC由於須要對整個塊進行回收,因此比Scavenge GC要慢,所以應該儘量減小Full GC的次數。在對JVM調優的過程當中,很大一部分工做就是對於FullGC的調節。有以下緣由可能致使Full GC:

· 年老代(Tenured)被寫滿

· 持久代(Perm)被寫滿

· System.gc()被顯示調用

·上一次GC以後Heap的各域分配策略動態變化

相關文章
相關標籤/搜索