JDK11 | 第六篇 : Epsilon 垃圾收集器

文章首發於公衆號《程序員果果》java

地址 : https://mp.weixin.qq.com/s/RhGXJImhp7Xm-wDrpPomkQ程序員

1、簡介

Epsilon(A No-Op Garbage Collector)垃圾回收器控制內存分配,可是不執行任何垃圾回收工做。一旦java的堆被耗盡,jvm就直接關閉。設計的目的是提供一個徹底消極的GC實現,分配有限的內存分配,最大限度下降消費內存佔用量和內存吞吐時的延遲時間。一個好的實現是隔離代碼變化,不影響其餘GC,最小限度的改變其餘的JVM代碼。微信

2、使用場景

  • Performance testing,什麼都不執行的GC很是適合用於差別性分析。no-op GC能夠用於過濾掉GC誘發的新能損耗,好比GC線程的調度,GC屏障的消耗,GC週期的不合適觸發,內存位置變化等。此外有些延遲者不是因爲GC引發的,好比scheduling hiccups, compiler transition hiccups,因此去除GC引起的延遲有助於統計這些延遲。
  • Memory pressure testing, 在測試java代碼時,肯定分配內存的閾值有助於設置內存壓力常量值。這時no-op就頗有用,它能夠簡單地接受一個分配的內存分配上限,當內存超限時就失敗。例如:測試須要分配小於1G的內存,就使用-Xmx1g參數來配置no-op GC,而後當內存耗盡的時候就直接crash。
  • VM interface testing, 以VM開發視角,有一個簡單的GC實現,有助於理解VM-GC的最小接口實現。它也用於證實VM-GC接口的健全性。
  • Extremely short lived jobs, 一個短聲明週期的工做可能會依賴快速退出來釋放資源,這個時候接收GC週期來清理heap實際上是在浪費時間,由於heap會在退出時清理。而且GC週期可能會佔用一會時間,由於它依賴heap上的數據量。
  • Last-drop latency improvements, 對那些極端延遲敏感的應用,開發者十分清楚內存佔用,或者是幾乎沒有垃圾回收的應用,此時耗時較長的GC週期將會是一件壞事。
  • Last-drop throughput improvements, 即使對那些無需內存分配的工做,選擇一個GC意味着選擇了一系列的GC屏障,全部的OpenJDK GC都是分代的,因此他們至少會有一個寫屏障。避免這些屏障能夠帶來一點點的吞吐量提高。

3、案例

使用G1垃圾收集器

代碼:jvm

public class TestEpsilon {

    public static void main(String[] args) {
        System.out.println("程序開始");
        boolean flag = true;
        List<Garbage> list = new ArrayList<>();
        long count = 0;
        while (flag) {
            list.add(new Garbage(list.size() + 1));
            if (list.size() == 1000000 && count == 0) {
                list.clear();
                count++;
            }
        }
        System.out.println("程序結束");
    }
}

class Garbage {

    private int number;

    public Garbage(int number) {
        this.number = number;
    }

    /**
     * GC在清除對象時,會調用finalize()方法
     */
    @Override
    public void finalize() {
        System.out.println(this + " : " + number + " is dying");
    }

    public int getNumber() {
        return number;
    }

    public void setNumber(int number) {
        this.number = number;
    }

}

啓動參數:ide

-Xms100m -Xmx100m

運行程序後,結果以下:測試

程序開始
...
com.gf.demo8.Garbage@15ddf76b : 305097 is dying
com.gf.demo8.Garbage@35e52705 : 305224 is dying
com.gf.demo8.Garbage@32c14bc1 : 305362 is dying
com.gf.demo8.Garbage@7521660a : 305705 is dying
com.gf.demo8.Garbage@f3da16a : 305948 is dying
com.gf.demo8.Garbage@13fc7287 : 306089 is dying
    at java.base/java.lang.ref.Finalizer.register(Finalizer.java:66)
    at java.base/java.lang.Object.<init>(Object.java:50)
    at com.gf.demo8.Garbage.<init>(TestEpsilon.java:28)
    at com.gf.demo8.TestEpsilon.main(TestEpsilon.java:14)
...

會發現G1一直回收對象,直到內存不夠用。this

使用Epsilon垃圾收集器

啓動參數:spa

UnlockExperimentalVMOptions:解鎖隱藏的虛擬機參數。線程

-XX:+UnlockExperimentalVMOptions -XX:+UseEpsilonGC -Xms100m -Xmx100m

運行程序後,結果以下:設計

程序開始
Terminating due to java.lang.OutOfMemoryError: Java heap space

會發現很快就內存溢出了,由於Epsilon不會去回收對象。

關注我


歡迎掃碼或微信搜索公衆號《程序員果果》關注我,更多精彩內容不能錯過 ~~

(轉載本站文章請註明做者和出處 程序員果果的博客

相關文章
相關標籤/搜索