| 主要特點 應(yīng)用場景 設(shè)置參數(shù) 基本運行原理   
       在《Java虛擬機垃圾回收(一) 基礎(chǔ)》中了解到如何判斷對象是存活還是已經(jīng)死亡?在《Java虛擬機垃圾回收(二) 垃圾回收算法》了解到Java虛擬機垃圾回收的幾種常見算法。        下面先來了解HotSpot虛擬機中的7種垃圾收集器:Serial、ParNew、Parallel Scavenge、Serial Old、Parallel Old、CMS、G1,先介紹一些垃圾收集的相關(guān)概念,再介紹它們的主要特點、應(yīng)用場景、以及一些設(shè)置參數(shù)和基本運行原理。 1、垃圾收集器概述
       垃圾收集器是垃圾回收算法(標記-清除算法、復制算法、標記-整理算法、火車算法)的具體實現(xiàn),不同商家、不同版本的JVM所提供的垃圾收集器可能會有很在差別,本文主要介紹HotSpot虛擬機中的垃圾收集器。 1-1、垃圾收集器組合
       JDK7/8后,HotSpot虛擬機所有收集器及組合(連線),如下圖: 
 
(A)、圖中展示了7種不同分代的收集器:        Serial、ParNew、Parallel Scavenge、Serial Old、Parallel Old、CMS、G1; (B)、而它們所處區(qū)域,則表明其是屬于新生代收集器還是老年代收集器:       新生代收集器:Serial、ParNew、Parallel Scavenge;       老年代收集器:Serial Old、Parallel Old、CMS;       整堆收集器:G1; (C)、兩個收集器間有連線,表明它們可以搭配使用:        Serial/Serial Old、Serial/CMS、ParNew/Serial Old、ParNew/CMS、Parallel Scavenge/Serial Old、Parallel Scavenge/Parallel Old、G1; (D)、其中Serial Old作為CMS出現(xiàn)"Concurrent Mode Failure"失敗的后備預案(后面介紹); 1-2、并發(fā)垃圾收集和并行垃圾收集的區(qū)別
(A)、并行(Parallel)        指多條垃圾收集線程并行工作,但此時用戶線程仍然處于等待狀態(tài);        如ParNew、Parallel Scavenge、Parallel Old; (B)、并發(fā)(Concurrent)        指用戶線程與垃圾收集線程同時執(zhí)行(但不一定是并行的,可能會交替執(zhí)行);       用戶程序在繼續(xù)運行,而垃圾收集程序線程運行于另一個CPU上;            如CMS、G1(也有并行); 1-3、Minor GC和Full GC的區(qū)別
(A)、Minor GC        又稱新生代GC,指發(fā)生在新生代的垃圾收集動作;        因為Java對象大多是朝生夕滅,所以Minor GC非常頻繁,一般回收速度也比較快; (B)、Full GC        又稱Major GC或老年代GC,指發(fā)生在老年代的GC;        出現(xiàn)Full GC經(jīng)常會伴隨至少一次的Minor GC(不是絕對,Parallel Sacvenge收集器就可以選擇設(shè)置Major GC策略);       Major GC速度一般比Minor GC慢10倍以上;          
下面將介紹這些收集器的特性、基本原理和使用場景,并重點分析CMS和G1這兩款相對復雜的收集器;但需要明確一個觀點:        沒有最好的收集器,更沒有萬能的收集;       選擇的只能是適合具體應(yīng)用場景的收集器。 2、Serial收集器
       Serial(串行)垃圾收集器是最基本、發(fā)展歷史最悠久的收集器;        JDK1.3.1前是HotSpot新生代收集的唯一選擇; 1、特點       針對新生代;       采用復制算法;       單線程收集;        進行垃圾收集時,必須暫停所有工作線程,直到完成;                    即會"Stop The World";       Serial/Serial Old組合收集器運行示意圖如下: 
 
2、應(yīng)用場景       依然是HotSpot在Client模式下默認的新生代收集器;       也有優(yōu)于其他收集器的地方: 
      簡單高效(與其他收集器的單線程相比);       對于限定單個CPU的環(huán)境來說,Serial收集器沒有線程交互(切換)開銷,可以獲得最高的單線程收集效率;       在用戶的桌面應(yīng)用場景中,可用內(nèi)存一般不大(幾十M至一兩百M),可以在較短時間內(nèi)完成垃圾收集(幾十MS至一百多MS),只要不頻繁發(fā)生,這是可以接受的 3、設(shè)置參數(shù)       "-XX:+UseSerialGC":添加該參數(shù)來顯式的使用串行垃圾收集器; 4、Stop TheWorld說明       JVM在后臺自動發(fā)起和自動完成的,在用戶不可見的情況下,把用戶正常的工作線程全部停掉,即GC停頓;       會帶給用戶不良的體驗; 
      從JDK1.3到現(xiàn)在,從Serial收集器-》Parallel收集器-》CMS-》G1,用戶線程停頓時間不斷縮短,但仍然無法完全消除;       更多"Stop The World"信息請參考:《Java虛擬機垃圾回收(一) 基礎(chǔ)》"2-2、可達性分析算法" 更多Serial收集器請參考:       《Memory Management in the Java HotSpot? Virtual Machine》 4.3節(jié) Serial Collector(內(nèi)存管理白皮書):http://www.oracle.com/technetwork/java/javase/tech/memorymanagement-whitepaper-1-150020.pdf       《Java Platform, Standard Edition HotSpot Virtual Machine Garbage Collection Tuning Guide》 第5節(jié) Available Collectors(官方的垃圾收集調(diào)優(yōu)指南):http://docs.oracle.com/javase/8/docs/technotes/guides/vm/gctuning/collectors.html#sthref27 3、ParNew收集器
      ParNew垃圾收集器是Serial收集器的多線程版本。 1、特點       除了多線程外,其余的行為、特點和Serial收集器一樣;       如Serial收集器可用控制參數(shù)、收集算法、Stop The World、內(nèi)存分配規(guī)則、回收策略等;       兩個收集器共用了不少代碼;       ParNew/Serial Old組合收集器運行示意圖如下: 
 
2、應(yīng)用場景       在Server模式下,ParNew收集器是一個非常重要的收集器,因為除Serial外,目前只有它能與CMS收集器配合工作;       但在單個CPU環(huán)境中,不會比Serail收集器有更好的效果,因為存在線程交互開銷。 3、設(shè)置參數(shù)       "-XX:+UseConcMarkSweepGC":指定使用CMS后,會默認使用ParNew作為新生代收集器;       "-XX:+UseParNewGC":強制指定使用ParNew;           "-XX:ParallelGCThreads":指定垃圾收集的線程數(shù)量,ParNew默認開啟的收集線程與CPU的數(shù)量相同; 4、為什么只有ParNew能與CMS收集器配合       CMS是HotSpot在JDK1.5推出的第一款真正意義上的并發(fā)(Concurrent)收集器,第一次實現(xiàn)了讓垃圾收集線程與用戶線程(基本上)同時工作;       CMS作為老年代收集器,但卻無法與JDK1.4已經(jīng)存在的新生代收集器Parallel Scavenge配合工作;       因為Parallel Scavenge(以及G1)都沒有使用傳統(tǒng)的GC收集器代碼框架,而另外獨立實現(xiàn);而其余幾種收集器則共用了部分的框架代碼;       關(guān)于CMS收集器后面會詳細介紹。 4、Parallel Scavenge收集器
      Parallel Scavenge垃圾收集器因為與吞吐量關(guān)系密切,也稱為吞吐量收集器(Throughput Collector)。 1、特點 
(A)、有一些特點與ParNew收集器相似       新生代收集器;       采用復制算法;       多線程收集; (B)、主要特點是:它的關(guān)注點與其他收集器不同       CMS等收集器的關(guān)注點是盡可能地縮短垃圾收集時用戶線程的停頓時間;       而Parallel Scavenge收集器的目標則是達一個可控制的吞吐量(Throughput);       關(guān)于吞吐量與收集器關(guān)注點說明詳見本節(jié)后面; 
2、應(yīng)用場景       高吞吐量為目標,即減少垃圾收集時間,讓用戶代碼獲得更長的運行時間;       當應(yīng)用程序運行在具有多個CPU上,對暫停時間沒有特別高的要求時,即程序主要在后臺進行計算,而不需要與用戶進行太多交互;       例如,那些執(zhí)行批量處理、訂單處理、工資支付、科學計算的應(yīng)用程序; 3、設(shè)置參數(shù)       Parallel Scavenge收集器提供兩個參數(shù)用于精確控制吞吐量: 
(A)、"-XX:MaxGCPauseMillis"       控制最大垃圾收集停頓時間,大于0的毫秒數(shù);       MaxGCPauseMillis設(shè)置得稍小,停頓時間可能會縮短,但也可能會使得吞吐量下降;       因為可能導致垃圾收集發(fā)生得更頻繁; (B)、"-XX:GCTimeRatio"       設(shè)置垃圾收集時間占總時間的比率,0<n<100的整數(shù);       GCTimeRatio相當于設(shè)置吞吐量大小;       垃圾收集執(zhí)行時間占應(yīng)用程序執(zhí)行時間的比例的計算方法是: 
      1 / (1 + n)       例如,選項-XX:GCTimeRatio=19,設(shè)置了垃圾收集時間占總時間的5%--1/(1+19);       默認值是1%--1/(1+99),即n=99; 
垃圾收集所花費的時間是年輕一代和老年代收集的總時間; 如果沒有滿足吞吐量目標,則增加代的內(nèi)存大小以盡量增加用戶程序運行的時間;       此外,還有一個值得關(guān)注的參數(shù): 
(C)、"-XX:+UseAdptiveSizePolicy"       開啟這個參數(shù)后,就不用手工指定一些細節(jié)參數(shù),如: 
      新生代的大?。?Xmn)、Eden與Survivor區(qū)的比例(-XX:SurvivorRation)、晉升老年代的對象年齡(-XX:PretenureSizeThreshold)等;       JVM會根據(jù)當前系統(tǒng)運行情況收集性能監(jiān)控信息,動態(tài)調(diào)整這些參數(shù),以提供最合適的停頓時間或最大的吞吐量,這種調(diào)節(jié)方式稱為GC自適應(yīng)的調(diào)節(jié)策略(GC Ergonomiscs);     
      這是一種值得推薦的方式: 
      (1)、只需設(shè)置好內(nèi)存數(shù)據(jù)大?。ㄈ?-Xmx"設(shè)置最大堆);       (2)、然后使用"-XX:MaxGCPauseMillis"或"-XX:GCTimeRatio"給JVM設(shè)置一個優(yōu)化目標;       (3)、那些具體細節(jié)參數(shù)的調(diào)節(jié)就由JVM自適應(yīng)完成;               這也是Parallel Scavenge收集器與ParNew收集器一個重要區(qū)別;           更多目標調(diào)優(yōu)和GC自適應(yīng)的調(diào)節(jié)策略說明請參考:             
      《Memory Management in the Java HotSpot? Virtual Machine》 5節(jié) Ergonomics -- Automatic Selections and Behavior Tuning:http://www.oracle.com/technetwork/java/javase/tech/memorymanagement-whitepaper-1-150020.pdf       《Java Platform, Standard Edition HotSpot Virtual Machine Garbage Collection Tuning Guide》 第2節(jié) Ergonomics:http://docs.oracle.com/javase/8/docs/technotes/guides/vm/gctuning/ergonomics.html#ergonomics 
4、吞吐量與收集器關(guān)注點說明 
(A)、吞吐量(Throughput)       CPU用于運行用戶代碼的時間與CPU總消耗時間的比值;       即吞吐量=運行用戶代碼時間/(運行用戶代碼時間+垃圾收集時間);           高吞吐量即減少垃圾收集時間,讓用戶代碼獲得更長的運行時間; (B)、垃圾收集器期望的目標(關(guān)注點) 
(1)、停頓時間           停頓時間越短就適合需要與用戶交互的程序;       良好的響應(yīng)速度能提升用戶體驗; (2)、吞吐量       高吞吐量則可以高效率地利用CPU時間,盡快完成運算的任務(wù);       主要適合在后臺計算而不需要太多交互的任務(wù); (3)、覆蓋區(qū)(Footprint)       在達到前面兩個目標的情況下,盡量減少堆的內(nèi)存空間;       可以獲得更好的空間局部性; 更多Parallel Scavenge收集器的信息請參考:       官方的垃圾收集調(diào)優(yōu)指南 第6節(jié):http://docs.oracle.com/javase/8/docs/technotes/guides/vm/gctuning/parallel.html#parallel_collector   上面介紹的都是新生代收集器,接下來開始介紹老年代收集器;
 5、Serial Old收集器
      Serial Old是 Serial收集器的老年代版本; 1、特點       針對老年代;       采用"標記-整理"算法(還有壓縮,Mark-Sweep-Compact);       單線程收集;       Serial/Serial Old收集器運行示意圖如下: 
 
2、應(yīng)用場景       主要用于Client模式;       而在Server模式有兩大用途: 
      (A)、在JDK1.5及之前,與Parallel Scavenge收集器搭配使用(JDK1.6有Parallel Old收集器可搭配);       (B)、作為CMS收集器的后備預案,在并發(fā)收集發(fā)生Concurrent Mode Failure時使用(后面詳解); 更多Serial Old收集器信息請參考:       內(nèi)存管理白皮書 4.3.2節(jié):http://www.oracle.com/technetwork/java/javase/tech/memorymanagement-whitepaper-1-150020.pdf 6、Parallel Old收集器
      Parallel Old垃圾收集器是Parallel Scavenge收集器的老年代版本;       JDK1.6中才開始提供; 1、特點       針對老年代;       采用"標記-整理"算法;       多線程收集;       Parallel Scavenge/Parallel Old收集器運行示意圖如下: 
 
2、應(yīng)用場景       JDK1.6及之后用來代替老年代的Serial Old收集器;       特別是在Server模式,多CPU的情況下;       這樣在注重吞吐量以及CPU資源敏感的場景,就有了Parallel Scavenge加Parallel Old收集器的"給力"應(yīng)用組合; 3、設(shè)置參數(shù)       "-XX:+UseParallelOldGC":指定使用Parallel Old收集器; 更多Parallel Old收集器收集過程介紹請參考:       《內(nèi)存管理白皮書》 4.5.2節(jié):        http://www.oracle.com/technetwork/java/javase/tech/memorymanagement-whitepaper-1-150020.pdf 7、CMS收集器
      并發(fā)標記清理(Concurrent Mark Sweep,CMS)收集器也稱為并發(fā)低停頓收集器(Concurrent Low Pause Collector)或低延遲(low-latency)垃圾收集器;       在前面ParNew收集器曾簡單介紹過其特點; 1、特點       針對老年代;       基于"標記-清除"算法(不進行壓縮操作,產(chǎn)生內(nèi)存碎片);                   以獲取最短回收停頓時間為目標;       并發(fā)收集、低停頓;       需要更多的內(nèi)存(看后面的缺點);                    是HotSpot在JDK1.5推出的第一款真正意義上的并發(fā)(Concurrent)收集器;       第一次實現(xiàn)了讓垃圾收集線程與用戶線程(基本上)同時工作; 2、應(yīng)用場景       與用戶交互較多的場景;               希望系統(tǒng)停頓時間最短,注重服務(wù)的響應(yīng)速度;       以給用戶帶來較好的體驗;       如常見WEB、B/S系統(tǒng)的服務(wù)器上的應(yīng)用; 
3、設(shè)置參數(shù)       "-XX:+UseConcMarkSweepGC":指定使用CMS收集器; 4、CMS收集器運作過程       比前面幾種收集器更復雜,可以分為4個步驟: 
(A)、初始標記(CMS initial mark)       僅標記一下GC Roots能直接關(guān)聯(lián)到的對象;       速度很快;       但需要"Stop The World"; (B)、并發(fā)標記(CMS concurrent mark)       進行GC Roots Tracing的過程;       剛才產(chǎn)生的集合中標記出存活對象;       應(yīng)用程序也在運行;       并不能保證可以標記出所有的存活對象; (C)、重新標記(CMS remark)       為了修正并發(fā)標記期間因用戶程序繼續(xù)運作而導致標記變動的那一部分對象的標記記錄;       需要"Stop The World",且停頓時間比初始標記稍長,但遠比并發(fā)標記短;       采用多線程并行執(zhí)行來提升效率; (D)、并發(fā)清除(CMS concurrent sweep)       回收所有的垃圾對象; 
      整個過程中耗時最長的并發(fā)標記和并發(fā)清除都可以與用戶線程一起工作;       所以總體上說,CMS收集器的內(nèi)存回收過程與用戶線程一起并發(fā)執(zhí)行;       CMS收集器運行示意圖如下: 
           5、CMS收集器3個明顯的缺點 
                     (A)、對CPU資源非常敏感 
      并發(fā)收集雖然不會暫停用戶線程,但因為占用一部分CPU資源,還是會導致應(yīng)用程序變慢,總吞吐量降低。       CMS的默認收集線程數(shù)量是=(CPU數(shù)量+3)/4;       當CPU數(shù)量多于4個,收集線程占用的CPU資源多于25%,對用戶程序影響可能較大;不足4個時,影響更大,可能無法接受。         增量式并發(fā)收集器: 
      針對這種情況,曾出現(xiàn)了"增量式并發(fā)收集器"(Incremental Concurrent Mark Sweep/i-CMS);       類似使用搶占式來模擬多任務(wù)機制的思想,讓收集線程和用戶線程交替運行,減少收集線程運行時間;       但效果并不理想,JDK1.6后就官方不再提倡用戶使用。 更多請參考:       官方的《垃圾收集調(diào)優(yōu)指南》8.8節(jié) Incremental Mode:http://docs.oracle.com/javase/8/docs/technotes/guides/vm/gctuning/cms.html#CJAGIIEJ       《內(nèi)存管理白皮書》 4.6.3節(jié)可以看到一些描述; 
(B)、無法處理浮動垃圾,可能出現(xiàn)"Concurrent Mode Failure"失敗 
(1)、浮動垃圾(Floating Garbage)       在并發(fā)清除時,用戶線程新產(chǎn)生的垃圾,稱為浮動垃圾;       這使得并發(fā)清除時需要預留一定的內(nèi)存空間,不能像其他收集器在老年代幾乎填滿再進行收集;       也要可以認為CMS所需要的空間比其他垃圾收集器大;       "-XX:CMSInitiatingOccupancyFraction":設(shè)置CMS預留內(nèi)存空間;       JDK1.5默認值為68%;       JDK1.6變?yōu)榇蠹s92%;                
(2)、"Concurrent Mode Failure"失敗       如果CMS預留內(nèi)存空間無法滿足程序需要,就會出現(xiàn)一次"Concurrent Mode Failure"失??;       這時JVM啟用后備預案:臨時啟用Serail Old收集器,而導致另一次Full GC的產(chǎn)生;       這樣的代價是很大的,所以CMSInitiatingOccupancyFraction不能設(shè)置得太大。 
(C)、產(chǎn)生大量內(nèi)存碎片       由于CMS基于"標記-清除"算法,清除后不進行壓縮操作;       前面《Java虛擬機垃圾回收(二) 垃圾回收算法》"標記-清除"算法介紹時曾說過: 
      產(chǎn)生大量不連續(xù)的內(nèi)存碎片會導致分配大內(nèi)存對象時,無法找到足夠的連續(xù)內(nèi)存,從而需要提前觸發(fā)另一次Full GC動作。       解決方法:                 
(1)、"-XX:+UseCMSCompactAtFullCollection"       使得CMS出現(xiàn)上面這種情況時不進行Full GC,而開啟內(nèi)存碎片的合并整理過程;       但合并整理過程無法并發(fā),停頓時間會變長;       默認開啟(但不會進行,結(jié)合下面的CMSFullGCsBeforeCompaction); (2)、"-XX:+CMSFullGCsBeforeCompaction"       設(shè)置執(zhí)行多少次不壓縮的Full GC后,來一次壓縮整理;       為減少合并整理過程的停頓時間;       默認為0,也就是說每次都執(zhí)行Full GC,不會進行壓縮整理; 
      由于空間不再連續(xù),CMS需要使用可用"空閑列表"內(nèi)存分配方式,這比簡單實用"碰撞指針"分配內(nèi)存消耗大;       更多關(guān)于內(nèi)存分配方式請參考:《Java對象在Java虛擬機中的創(chuàng)建過程》       總體來看,與Parallel Old垃圾收集器相比,CMS減少了執(zhí)行老年代垃圾收集時應(yīng)用暫停的時間;       但卻增加了新生代垃圾收集時應(yīng)用暫停的時間、降低了吞吐量而且需要占用更大的堆空間; 
更多CMS收集器信息請參考:       《垃圾收集調(diào)優(yōu)指南》 8節(jié) Concurrent Mark Sweep (CMS) Collector:http://docs.oracle.com/javase/8/docs/technotes/guides/vm/gctuning/cms.html#concurrent_mark_sweep_cms_collector       《內(nèi)存管理白皮書》 4.6節(jié) Concurrent Mark-Sweep (CMS) Collector:http://www.oracle.com/technetwork/java/javase/tech/memorymanagement-whitepaper-1-150020.pdf 8、G1收集器
      G1(Garbage-First)是JDK7-u4才推出商用的收集器; 1、特點 
(A)、并行與并發(fā)       能充分利用多CPU、多核環(huán)境下的硬件優(yōu)勢;       可以并行來縮短"Stop The World"停頓時間;       也可以并發(fā)讓垃圾收集與用戶程序同時進行; (B)、分代收集,收集范圍包括新生代和老年代           能獨立管理整個GC堆(新生代和老年代),而不需要與其他收集器搭配;       能夠采用不同方式處理不同時期的對象;                        雖然保留分代概念,但Java堆的內(nèi)存布局有很大差別;       將整個堆劃分為多個大小相等的獨立區(qū)域(Region);       新生代和老年代不再是物理隔離,它們都是一部分Region(不需要連續(xù))的集合;       更多G1內(nèi)存布局信息請參考: 
      《垃圾收集調(diào)優(yōu)指南》 9節(jié):http://docs.oracle.com/javase/8/docs/technotes/guides/vm/gctuning/g1_gc.html#garbage_first_garbage_collection 
(C)、結(jié)合多種垃圾收集算法,空間整合,不產(chǎn)生碎片       從整體看,是基于標記-整理算法;       從局部(兩個Region間)看,是基于復制算法;       這是一種類似火車算法的實現(xiàn);         都不會產(chǎn)生內(nèi)存碎片,有利于長時間運行; (D)、可預測的停頓:低停頓的同時實現(xiàn)高吞吐量       G1除了追求低停頓處,還能建立可預測的停頓時間模型;       可以明確指定M毫秒時間片內(nèi),垃圾收集消耗的時間不超過N毫秒; 
2、應(yīng)用場景       面向服務(wù)端應(yīng)用,針對具有大內(nèi)存、多處理器的機器;       最主要的應(yīng)用是為需要低GC延遲,并具有大堆的應(yīng)用程序提供解決方案;       如:在堆大小約6GB或更大時,可預測的暫停時間可以低于0.5秒;                    用來替換掉JDK1.5中的CMS收集器;       在下面的情況時,使用G1可能比CMS好: 
      (1)、超過50%的Java堆被活動數(shù)據(jù)占用;       (2)、對象分配頻率或年代提升頻率變化很大;       (3)、GC停頓時間過長(長于0.5至1秒)。 
      是否一定采用G1呢?也未必: 
      如果現(xiàn)在采用的收集器沒有出現(xiàn)問題,不用急著去選擇G1;       如果應(yīng)用程序追求低停頓,可以嘗試選擇G1;       是否代替CMS需要實際場景測試才知道。 
3、設(shè)置參數(shù)       "-XX:+UseG1GC":指定使用G1收集器;       "-XX:InitiatingHeapOccupancyPercent":當整個Java堆的占用率達到參數(shù)值時,開始并發(fā)標記階段;默認為45;       "-XX:MaxGCPauseMillis":為G1設(shè)置暫停時間目標,默認值為200毫秒;       "-XX:G1HeapRegionSize":設(shè)置每個Region大小,范圍1MB到32MB;目標是在最小Java堆時可以擁有約2048個Region;       更多關(guān)于G1參數(shù)設(shè)置請參考: 
      《垃圾收集調(diào)優(yōu)指南》 10.5節(jié):http://docs.oracle.com/javase/8/docs/technotes/guides/vm/gctuning/g1_gc_tuning.html#important_defaults 4、為什么G1收集器可以實現(xiàn)可預測的停頓       G1可以建立可預測的停頓時間模型,是因為: 
      可以有計劃地避免在Java堆的進行全區(qū)域的垃圾收集;       G1跟蹤各個Region獲得其收集價值大小,在后臺維護一個優(yōu)先列表;       每次根據(jù)允許的收集時間,優(yōu)先回收價值最大的Region(名稱Garbage-First的由來);       這就保證了在有限的時間內(nèi)可以獲取盡可能高的收集效率; 
5、一個對象被不同區(qū)域引用的問題       一個Region不可能是孤立的,一個Region中的對象可能被其他任意Region中對象引用,判斷對象存活時,是否需要掃描整個Java堆才能保證準確?       在其他的分代收集器,也存在這樣的問題(而G1更突出): 
      回收新生代也不得不同時掃描老年代?       這樣的話會降低Minor GC的效率;       解決方法: 
      無論G1還是其他分代收集器,JVM都是使用Remembered Set來避免全局掃描: 
      每個Region都有一個對應(yīng)的Remembered Set;       每次Reference類型數(shù)據(jù)寫操作時,都會產(chǎn)生一個Write Barrier暫時中斷操作;       然后檢查將要寫入的引用指向的對象是否和該Reference類型數(shù)據(jù)在不同的Region(其他收集器:檢查老年代對象是否引用了新生代對象);       如果不同,通過CardTable把相關(guān)引用信息記錄到引用指向?qū)ο蟮乃赗egion對應(yīng)的Remembered Set中;                      
      當進行垃圾收集時,在GC根節(jié)點的枚舉范圍加入Remembered Set;       就可以保證不進行全局掃描,也不會有遺漏。 
6、G1收集器運作過程       不計算維護Remembered Set的操作,可以分為4個步驟(與CMS較為相似)。 
(A)、初始標記(Initial Marking)       僅標記一下GC Roots能直接關(guān)聯(lián)到的對象;       且修改TAMS(Next Top at Mark Start),讓下一階段并發(fā)運行時,用戶程序能在正確可用的Region中創(chuàng)建新對象;       需要"Stop The World",但速度很快; (B)、并發(fā)標記(Concurrent Marking)       進行GC Roots Tracing的過程;       剛才產(chǎn)生的集合中標記出存活對象;       耗時較長,但應(yīng)用程序也在運行;       并不能保證可以標記出所有的存活對象; (C)、最終標記(Final Marking)       為了修正并發(fā)標記期間因用戶程序繼續(xù)運作而導致標記變動的那一部分對象的標記記錄;       上一階段對象的變化記錄在線程的Remembered Set Log;       這里把Remembered Set Log合并到Remembered Set中;                            需要"Stop The World",且停頓時間比初始標記稍長,但遠比并發(fā)標記短;       采用多線程并行執(zhí)行來提升效率; (D)、篩選回收(Live Data Counting and Evacuation)       首先排序各個Region的回收價值和成本;       然后根據(jù)用戶期望的GC停頓時間來制定回收計劃;       最后按計劃回收一些價值高的Region中垃圾對象;                            回收時采用"復制"算法,從一個或多個Region復制存活對象到堆上的另一個空的Region,并且在此過程中壓縮和釋放內(nèi)存;       可以并發(fā)進行,降低停頓時間,并增加吞吐量; 
      G1收集器運行示意圖如下: 
 
更多G1收集器信息請參考:       《垃圾收集調(diào)優(yōu)指南》 9節(jié) Garbage-First Garbage Collector:http://docs.oracle.com/javase/8/docs/technotes/guides/vm/gctuning/g1_gc.html#garbage_first_garbage_collection       《垃圾收集調(diào)優(yōu)指南》 10節(jié) Garbage-First Garbage Collector Tuning:http://docs.oracle.com/javase/8/docs/technotes/guides/vm/gctuning/g1_gc_tuning.html#g1_gc_tuning          
      到這里,我們大體了解HotSpot虛擬機中的所有垃圾收集器,后面我們將去了解JVM的一些內(nèi)存分配與回收策略、JVM垃圾收集相關(guān)調(diào)優(yōu)方法……   【參考資料】 1、《編譯原理》第二版 第7章 2、《深入理解Java虛擬機:JVM高級特性與最佳實踐》第二版 第3章 3、《The Java Virtual Machine Specification》Java SE 8 Edition:https://docs.oracle.com/javase/specs/jvms/se8/html/index.html 4、《Java Platform, Standard Edition HotSpot Virtual Machine Garbage Collection Tuning Guide》:http://docs.oracle.com/javase/8/docs/technotes/guides/vm/gctuning/index.html 5、《Memory Management in the Java HotSpot? Virtual Machine》:http://www.oracle.com/technetwork/java/javase/tech/memorymanagement-whitepaper-1-150020.pdf 6、HotSpot虛擬機參數(shù)官方說明:http://docs.oracle.com/javase/8/docs/technotes/tools/unix/java.html 7、《Thinking in Java》第四版 5.5 清理:終結(jié)處理和垃圾回收; |