隨著堆利用率接近 100% 并且不可避免的 OutOfMemoryError 可能導(dǎo)致生產(chǎn) JVM 崩潰,傾向于實(shí)時 JAVA 應(yīng)用程序的 DevOps 專業(yè)人員不會要求一種機(jī)制來強(qiáng)制 Java 垃圾收集并釋放幾兆字節(jié)的內(nèi)存。
不幸的是,這種立即釋放內(nèi)存的愿望肯定是徒勞的,因?yàn)樵?Java 中沒有辦法強(qiáng)制進(jìn)行垃圾收集 (GC)。但是,這里有五種策略可以讓 Java 虛擬機(jī) ( JVM ) 確定任務(wù)的優(yōu)先級。
1. 調(diào)用System.gc()
開發(fā)人員可以在其代碼中的任何位置調(diào)用 System.gc() 來指示 JVM 優(yōu)先考慮垃圾收集。當(dāng)開發(fā)人員調(diào)用此方法時——JVM 上沒有極端負(fù)載——Java GC 循環(huán)將在幾秒鐘內(nèi)發(fā)生。
System.gc();
2.調(diào)用Runtime.getRuntime().gc()
另一種選擇是使用Runtime.getRuntime().gc()調(diào)用。這是JDK為想要強(qiáng)制進(jìn)行 Java 垃圾收集的開發(fā)人員提供的第二個功能。Runtime.getRuntime().gc()調(diào)用實(shí)際上只是在后臺調(diào)用方法System.gc ( ) 。結(jié)果,這兩個方法調(diào)用是完全一樣的。
Runtime.getRuntime().gc();
3.使用jmap強(qiáng)制GC
Java 內(nèi)存映射 (JMAP) 實(shí)用程序有一個打印 Java 堆直方圖的方法。jmap命令的一個副作用是,當(dāng)它被調(diào)用時,它會強(qiáng)制執(zhí)行垃圾收集例程。但是,這并不是強(qiáng)制進(jìn)行 Java 垃圾收集的萬無一失的方法。如果 JVM 很忙并且無法執(zhí)行 GC 循環(huán),則該命令將出錯。
$ jmap -histo:live 7544
Figure 1 您可以嘗試使用 JDK 的 JMAP 和 JCMD 實(shí)用程序強(qiáng)制執(zhí)行 Java 垃圾收集。
4. 使用jcmd的命令行 Java GC
Java 診斷命令 (JCMD) 是另一個JDK 實(shí)用程序,如果 JVM 可以安全地安排 stop-the-world 暫停,它將觸發(fā)垃圾收集例程。如果不是,此命令將以與jmap實(shí)用程序相同的方式出錯。
$ jcmd 7544 GC.run
5. 使用 JConsole 或 Java Mission Control
JConsole 和 Java Mission Control 都提供了與 Java 診斷命令實(shí)用程序交互的用戶友好界面,可用于強(qiáng)制執(zhí)行 Java 垃圾收集。JConsole監(jiān)控工具在其內(nèi)存管理頁面上提供了一個按鈕,上面寫著Run Garbage Collection。Java Mission Control 允許開發(fā)人員選擇任何jcmd 開關(guān)——包括GC.run——并通過單擊按鈕執(zhí)行命令。
但是,這些工具在 Java 垃圾收集方面并沒有做任何獨(dú)特的事情。他們只是在幕后調(diào)用jcmd實(shí)用程序。
Figure 2 Java Mission Control 和 JConsole 都有可視化工具,允許您強(qiáng)制執(zhí)行 Java GC。
如何強(qiáng)制 Java 垃圾回收
雖然開發(fā)人員永遠(yuǎn)不能真正強(qiáng)制 Java 垃圾收集,但有一些方法可以讓 JVM 優(yōu)先考慮內(nèi)存管理功能。回顧一下,嘗試強(qiáng)制執(zhí)行 Java 垃圾收集的五種方法是:
- 調(diào)用System.gc()命令。
- 調(diào)用getRuntime().gc()命令。
- 使用jmap命令。
- 使用jcmd命令。
- 使用 JConsole 或 Java Mission Control。