JVM的垃圾回收機制真是通俗易懂

堆內存的劃分

分為三個部分(以下名詞表示同一個區):

  • 新生區、新生代、年輕代
  • 養老區、老年區、老年代
  • 永久區、永久代

在這裡插入圖片描述

劃分區域的目的

唯一目的就是優化GC性能。

如果沒有分代,我們所有的對象都放在一塊,GC的時候我們需要對堆的所有區域進行掃描。而很多的對象都是“朝生夕死”的,如果把創建的新的對象都放在某一地方,當GC的時候就先把“朝生夕死”對象的區域進行回收,這樣就會騰出很多大的空間來。

一、新生區的垃圾回收機制

新生區分為:Eden區、Survivor0區、Survivor1區(也稱為from區和to區)
其中Eden區占80%的內存空間,每塊Survivor各占用10%的內存空間(如:Eden占800M,每個Survivor占100M)

1.開始時創建的對象都是分配在Eden區域中,當Eden區快滿瞭,就會觸發垃圾回收Minor GC(使用復制算法進行垃圾回收)

在這裡插入圖片描述

2.Minor GC處理後,首先會把Eden區中還存活著的對象一次性轉入其中一塊空閑著的Survivor區。然後清空Eden區,之後創建的對象就繼續放入Eden區中瞭,直至下次Eden又被填滿。

在這裡插入圖片描述

3.Eden再次被填滿時,就會再次出發Minor GC,清理後(Minor會清理Eden區和Survivor區的內存),Eden區和存在對象的Survivor區(此時的from區)中存活的對象轉移到另一塊空著的Survivor區中(此時的to區),並清空Eden區和之前存在對象的Survivor區(此時變為to區瞭,“From”和“To”會交換他們的角色,也就是新的“To”就是上次GC前的“From”,新的“From”就是上次GC前的“To”。)

這就是復制算法的流程。
一直要保持一個Survivor區是空的以提供復制算法垃圾回收,而這塊區域的內存隻占整塊的10%,其他90%內存都可以被使用,課件內存利用率還是相當高的。

二、什麼時候進入老年區呢?

1 經歷15次GC後進入老年區

默認情況下,如果新生區中的某個對象經歷瞭15次GC後,還是沒有被回收掉,那麼它就會被轉入老年區。
可通過JVM參數“-XX:MaxTenuringThreshold”來設置,默認是15。

2 動態對象年齡判斷

這種方法不用等到經歷GC15次。
假如一批對象總大小大於當前Survivor區內存的50%,那麼大於等於這批對象年齡的對象就會被轉移到老年區。

例:假設Survivor0區中的兩個對象都經歷的3次GC(年齡3),而且這兩個對象總大小50M,超過瞭Survivor0區內存大小的一半。那麼此時Survivor0區中年齡大於等於3歲的對象就都要被全部轉移到老年區。

在這裡插入圖片描述

3 大對象直接進入老年代

有一個JVM參數"-XX:PretenureSizeThreshold",默認值是0,表示任何情況都先把對象分配給Eden區。
若設置為1048576字節,也就是1M。則表示當創建的對象大於1M時,就會直接把這個對象放入到老年區,就根本不會經過新生區瞭。
這麼做的原因:大對象在經歷復制算法進行GC的時候會降低性能。

4 Minor GC後存活的對象太多無法放入Survivor區瞭

Minor GC後存活的對象太多,導致Survivor區放不下瞭,此時就會將所有的對象直接轉移到老年區中。

三、老年區空間分配擔保原則

執行每一次Minor GC前,JVM都先檢查一下老年區可用的內存空間是否大於新生區所有對象的總大小。

原因:極端情況下,Minor GC後,新生代中所有的對象都活瞭下來,那就會把所有新生代中的對象放入老年區中。

  • 如果說老年區可用內存大於新生代對象總大小,那麼就可以放心的執行Minor GC。
  • 但如果老年區內存小於新生區對象的總大小,這時候就會看一個參數:“-XX:HandlePromotionFailure”是否設置為true瞭。如果為true,就進入下一次判斷,看老年區可用內存是否大於之前每次Minor GC後進入老年區對象的平均大小。如果老年代可用內存小於平均大小或是參數沒有設置成true,那就會直接觸發“Full GC”,就是對老年代進行垃圾回收,騰出空間後,再進行Minor GC,相當於對新生區、老年區統一做瞭一次清理。

三種情況遞進理解:

1.如果Minor GC後,存活的對象<Survivor區大小,直接進入Survivor區即可;

2.如果Minor GC後,存活的對象>Survivor區大小,但<老年區可用內存,直接進入老年區;

3.若Minor GC後,此時老年區都放不下這些存活的對象瞭,就會觸發Full GC;
如果Full GC後老年區內存還是不夠用,就會導致OOM內存溢出。

四、老年區垃圾回收算法

標記整理算法

【原理】

一開始對象都是任意分佈的,在經歷完垃圾回收之後,就會標記出哪些是存活對象,哪些是垃圾對象,然後就會把這些存活的對象在內存中進行整理移動,盡量都挪到一邊去靠在一起,然後再把垃圾對象進行清除,這樣做的好處就是避免瞭垃圾回收後產生的大片內存碎片。

在這裡插入圖片描述

【缺點】

較為耗時,比復制算法慢10倍;

所以如果系統頻繁出現Full GC,會嚴重影響系統性能,出現卡頓。所以JVM優化的一大問題就是減少Full GC頻率。

五、垃圾回收器

新生區和老年區進行垃圾回收時是通過不同的垃圾回收器進行回收的

Seral 和 Seral Old垃圾回收器

  • 分別用於回收新生區和老年區。
  • 單線程運行,垃圾回收時會停止我們系統的其他線程,再執行垃圾回收(不再使用);

ParNew和CMS垃圾回收器

  • 分別用於新生區和老年區;
  • 多線程並發,性能更好,現在一般是線上生產系統的標配。

G1垃圾回收器

統一收集新生區和老年區,采用更加優秀的算法機制。

總結

本篇文章就到這裡瞭,希望能夠給你帶來幫助,也希望您能夠多多關註WalkonNet的更多內容!    

推薦閱讀: