Java 的七種垃圾收集器
了解 Java 中的內存管理。
用 C 或 C++ 這樣的編程語言寫一個應用時,需要編寫代碼來銷毀內存中不再需要的對象。當應用程序擴展得越來越複雜時,未使用對象被忽略釋放的可能性就越大。這會導致內存泄露,最終內存耗盡,在某個時刻將沒有更多的內存可以分配。結果就是應用程序運行失敗並出現 OutOfMemoryError 錯誤。但在 Java 中, 垃圾收集器 (GC)會在程序執行過程中自動運行,減輕了手動分配內存和可能的內存泄漏的任務。
垃圾收集器並不只有一種,Java 虛擬機(JVM)有七種不同的垃圾收集器,了解每種垃圾收集器的目的和優點是很有用的。
1、Serial 收集器
垃圾收集器的原始實現,使用單線程。當垃圾收集器運行時,會停止應用程序(通常稱為「stop the world」事件)。適用於能夠承受短暫停頓的應用程序。該垃圾收集器佔用內存空間比較小,因此這是嵌入式應用程序的首選垃圾收集器類型。在運行時使用以下命令啟用該垃圾收集器:
$ java -XX:+UseSerialGC
2、Parallel 收集器
像 Serial 收集器一樣,Parallel 收集器也使用「stop the world」方法。這意味著,當垃圾收集器運行時,應用程序線程會停止。但是不同的是,Parallel 收集器運行時有多個線程執行垃圾收集操作。這種類型的垃圾收集器適用於在多線程和多處理器環境中運行中到大型數據集的應用程序。
這是 JVM 中的默認垃圾收集器,也被稱為吞吐量收集器。使用該垃圾收集器時可以通過使用各種合適的 JVM 參數進行調優,例如吞吐量、暫停時間、線程數和內存佔用。如下:
- 線程數:
-XX:ParallelGCThreads=<N>
- 暫停時間:
-XX:MaxGCPauseMillis=<N>
- 吞吐量(垃圾收集花費的時間與實際應用程序執行的時間相比):
-XX:GCTimeRatio=<N>
- 最大堆內存:
-Xmx<N>
Parallel 收集器可以使用該命令顯式啟用:java -XX:+UseParallelGC
。使用這個命令,指定在新生代中通過多個線程進行垃圾回收,而老年代中的垃圾收集和內存壓縮仍使用單個線程完成的。
還有一個版本的的 Parallel 收集器叫做 「Parallel Old GC」,它對新生代和老年代都使用多線程,啟用命令如下:
$ java -XX:+UseParallelOldGC
3、Concurrent Mark Sweep(CMS)收集器
Concurrent Mark Sweep(CMS)垃圾收集器與應用程序並行運行。對於新生代和老年代都使用了多線程。在 CMS 垃圾收集器刪除無用對象後,不會對存活對象進行內存壓縮。該垃圾收集器和應用程序並行運行,會降低應用程序的響應時間,適用於停頓時間較短的應用程序。這個收集器在 Java8 已過時,並在 Java14 中被移除。如果你仍在使用有這個垃圾收集器的 Java 版本,可以使用如下命令啟用:
$ java -XX:+UseConcMarkSweepGC
在 CMS 垃圾收集器使用過程中,應用程序將暫停兩次。首次暫停發生在標記可直接訪問的存活對象時,這個暫停被稱為初始標記。第二次暫停發生在 CMS 收集器結束時期,來修正在並發標記過程中,應用程序線程在 CMS 垃圾回收完成後更新對象時被遺漏的對象。這就是所謂的重新標記。
4、G1 收集器
G1 垃圾收集器旨在替代 GMS。G1 垃圾收集器具備並行、並發以及增量壓縮,且暫停時間較短。與 CMS 收集器使用的內存布局不同,G1 收集器將堆內存劃分為大小相同的區域,通過多個線程觸發全局標記階段。標記階段完成後,G1 知道哪個區域可能大部分是空的,並首選該區域作為清除/刪除階段。
在 G1 收集器中,一個對象如果大小超過半個區域容量會被認為是一個「大對象」 。這些對象被放置在老年代中,在一個被稱為「humongous region」的區域中。 啟用 G1 收集器的命令如下:
$ java -XX:+UseG1GC
5、Epsilon 收集器
該垃圾收集器是在 Java11 中引入的,是一個 no-op(無操作)收集器。它不做任何實際的內存回收,只負責管理內存分配。Epsilon 只在當你知道應用程序的確切內存佔用情況並且不需要垃圾回收時使用。啟用命令如下:
$ java -XX:+UnlockExperimentalVMOptions -XX:+UseEpsilonGC
6、Shenandoah 收集器
Shenandoah 是在 JDK12 中引入的,是一種 CPU 密集型垃圾收集器。它會進行內存壓縮,立即刪除無用對象並釋放操作系統的空間。所有的這一切與應用程序線程並行發生。啟用命令如下:
$ java -XX:+UnlockExperimentalVMOptions -XX:+UseShenandoahGC
7、ZGC 收集器
ZGC 為低延遲需要和大量堆空間使用而設計,允許當垃圾回收器運行時 Java 應用程序繼續運行。ZGC 收集器在 JDK11 引入,在 JDK12 改進。在 JDK15,ZGC 和 Shenandoah 都被移出了實驗階段。啟用 ZGC 收集器使用如下命令:
$ java -XX:+UnlockExperimentalVMOptions -XX:+UseZGC
靈活的垃圾收集器
Java 為我們提供了靈活的內存管理方式,熟悉不同的可用方法有助於為正在開發或運行的應用程序選擇最合適的內存管理方式。
via: https://opensource.com/article/22/7/garbage-collection-java
作者:Jayashree Huttanagoudar 選題:lkxed 譯者:Veryzzj 校對:wxy
本文轉載來自 Linux 中國: https://github.com/Linux-CN/archive