我正在尝试找出 map 消耗的 RAM 量。所以,我做了以下事情;-
Map<String,Double> cr = crPair.collectAsMap(); // 200+ entries
System.out.println("free memory = " + Runtime.getRuntime().freeMemory());
Map<String,Double> gen = new HashMap<>();
gen.putAll(cr);
System.out.println("free memory = " + Runtime.getRuntime().freeMemory());
我得到的回复是:-
free memory = 940714880
free memory = 940714880
我基本上想看看在超过特定阈值之前我的 map 可以支持多少条目。然而,这意味着没有额外的字节。我错过了什么?此外,任何人都可以粗略估计 HashMap 占用的空间量(1000 个条目的顺序)。
最佳答案
Runtime.freeMemory
仅返回粗略估计。它不适合测量给定对象占用的内存。
您可以使用 JMH使用内置的 -prof gc
分析器以获得更准确的估计。这是一个示例基准,用于测量 HashMap 占用的空间(不计算键和值本身使用的内存)。
package bench;
import org.openjdk.jmh.annotations.*;
import java.util.HashMap;
import java.util.Map;
import java.util.concurrent.ThreadLocalRandom;
import java.util.stream.Collectors;
import java.util.stream.IntStream;
@State(Scope.Benchmark)
public class MapSizeBench {
@Param({"10", "100", "1000", "10000", "100000"})
int size;
Map<String, Double> map;
@Setup
public void setup() {
map = IntStream.range(0, size).boxed().collect(Collectors.toMap(
n -> "key" + n,
n -> ThreadLocalRandom.current().nextDouble()));
}
@Benchmark
public Map<String, Double> hashMap() {
return new HashMap<>(map);
}
}
gc.alloc.rate.norm
值将显示在执行带有 @Benchmark
注释标记的方法时分配的字节数:
Benchmark (size) Mode Cnt Score Error Units
MapSizeBench.hashMap:·gc.alloc.rate.norm 10 avgt 5 448,000 ± 0,001 B/op
MapSizeBench.hashMap:·gc.alloc.rate.norm 100 avgt 5 4288,001 ± 0,001 B/op
MapSizeBench.hashMap:·gc.alloc.rate.norm 1000 avgt 5 40256,009 ± 0,002 B/op
MapSizeBench.hashMap:·gc.alloc.rate.norm 10000 avgt 5 385640,093 ± 0,017 B/op
MapSizeBench.hashMap:·gc.alloc.rate.norm 100000 avgt 5 4248681,606 ± 0,329 B/op
也就是说,在启用 CompressedOops 的 JDK 8 x64 上,一个包含 1000 个元素的 HashMap 将占用大约 40KB。
或者,您可以进行完整的堆转储:
jmap -dump:format=b,file=heap.hprof <pid>
并在Eclipse Memory Analyzer中进行分析.该工具可以向您显示任何给定对象或对象图占用的内存。
关于使用的 Java 内存,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47389202/