1メガバイト以上のJavaバイト配列は二回RAMを占めます

ゲオルク:

Windowsの10の下のコードを実行している/ OpenJDKの11.0.4_x64は出力として生成used: 197expected usage: 200この手段は百万要素の200のバイト配列は、約取ること。200メガバイトのRAM。すべての罰金。

Iからコードのバイト配列の割り当てを変更したときnew byte[1000000]new byte[1048576](1024×1024要素に、である)、それが出力として生成used: 417し、expected usage: 200一体何?

import java.io.IOException;
import java.util.ArrayList;

public class Mem {
    private static Runtime rt = Runtime.getRuntime();
    private static long free() { return rt.maxMemory() - rt.totalMemory() + rt.freeMemory(); }
    public static void main(String[] args) throws InterruptedException, IOException {
        int blocks = 200;
        long initiallyFree = free();
        System.out.println("initially free: " + initiallyFree / 1000000);
        ArrayList<byte[]> data = new ArrayList<>();
        for (int n = 0; n < blocks; n++) { data.add(new byte[1000000]); }
        System.gc();
        Thread.sleep(2000);
        long remainingFree = free();
        System.out.println("remaining free: " + remainingFree / 1000000);
        System.out.println("used: " + (initiallyFree - remainingFree) / 1000000);
        System.out.println("expected usage: " + blocks);
        System.in.read();
    }
}

VisualVMのを少し深く見ると、私は予想通り最初のケースすべてに参照してください。

バイト配列は、200メガバイトを取ります

後者の場合は、バイト配列に加えて、私はバイト配列としてRAMの同じ量を取ってint型の配列の同じ数を参照してください。

int型の配列は、追加の200メガバイトを取ります

これらのint型の配列は、方法によって、それらが参照されていることを示していないが、私はごみ収集それらができない...(バイト配列は、それらが参照されているだけで罰金を示しています。)

ここで何が起こっている任意のアイデア?

drekbour:

これは説明のアウトオブボックスの動作ですG1のガベージコレクタ 1メガバイト「地域」にされ、一般的にデフォルトと他のGCで動作するJava 9にJVMのデフォルトになった有効にすると、さまざまな数字を提供します。

半分以上の領域サイズである任意のオブジェクトは、単にヒープ領域サイズの倍数よりもわずかに大きいオブジェクトの場合、この未使用領域は、ヒープが断片化する可能性があります...「でかい」と考えられています。

私は走ったjava -Xmx300M -XX:+PrintGCDetails、それはヒープがでかい領域によって排出される示しています。

[0.202s][info   ][gc,heap        ] GC(51) Old regions: 1->1
[0.202s][info   ][gc,heap        ] GC(51) Archive regions: 2->2
[0.202s][info   ][gc,heap        ] GC(51) Humongous regions: 296->296
[0.202s][info   ][gc             ] GC(51) Pause Full (G1 Humongous Allocation) 297M->297M(300M) 1.935ms
[0.202s][info   ][gc,cpu         ] GC(51) User=0.01s Sys=0.00s Real=0.00s
...
Exception in thread "main" java.lang.OutOfMemoryError: Java heap space

私たちは、1MiBがしたいbyte[]追加して「半分以下G1領域サイズ」であることを-XX:G1HeapRegionSize=4M、機能アプリケーションを提供します:

[0.161s][info   ][gc,heap        ] GC(19) Humongous regions: 0->0
[0.161s][info   ][gc,metaspace   ] GC(19) Metaspace: 320K->320K(1056768K)
[0.161s][info   ][gc             ] GC(19) Pause Full (System.gc()) 274M->204M(300M) 9.702ms
remaining free: 100
used: 209
expected usage: 200

G1の深さ概要:https://www.oracle.com/technical-resources/articles/java/g1gc.html

G1の破砕詳細:https://docs.oracle.com/en/java/javase/13/gctuning/garbage-first-garbage-collector-tuning.html#GUID-2428DA90-B93D-48E6-B336-A849ADF1C552

おすすめ

転載: http://43.154.161.224:23101/article/api/json?id=322674&siteId=1