Java 8 尽管 (Max)MetaspaceSize 为 Metaspace 保留最小 1G

Java 8 reserves minimum 1G for Metaspace despite (Max)MetaspaceSize

Java 8 刚启动就预留1G给Metaspace。这意味着最小元空间大小为 1G。但是我将 MetaspaceSize 设置为 300m,将 MaxMetaspaceSize 设置为 400m。为什么 Java 保留的比我允许的多?

Java版本

$ java -version
java version "1.8.0_45"
Java(TM) SE Runtime Environment (build 1.8.0_45-b14)
Java HotSpot(TM) 64-Bit Server VM (build 25.45-b02, mixed mode)

VM 标志

$ jcmd 21689 VM.flags
21689:
-XX:CICompilerCount=3 -XX:ConcGCThreads=1 -XX:G1HeapRegionSize=1048576 -XX:InitialHeapSize=62914560 -XX:+ManagementServer -XX:MarkStackSize=4194304 -XX:MaxHeapSize=1006632960 -XX:MaxMetaspaceSize=399998976 -XX:MaxNewSize=603979776 -XX:MetaspaceSize=299999232 -XX:MinHeapDeltaBytes=1048576 -XX:NativeMemoryTracking=summary -XX:+UseCompressedClassPointers -XX:+UseCompressedOops -XX:+UseFastUnorderedTimeStamps -XX:+UseG1GC 

NMT

[jetty9-proxy@bm01 bin]$ jcmd 21689 VM.native_memory
21689:

Native Memory Tracking:

Total: reserved=2769543KB, committed=1311159KB

-                     Class (reserved=1221904KB, committed=197904KB)
                            (classes #36543)
                            (malloc=3344KB #44041) 
                            (mmap: reserved=1218560KB, committed=194560KB) 

刚开始是

Total: reserved=2402748KB, committed=150796KB     

-                     Class (reserved=1056956KB, committed=7868KB)
                            (classes #1300)
                            (malloc=188KB #564) 
                            (mmap: reserved=1056768KB, committed=7680KB) 

Class (reserved=1221904KB

这不是正在使用的内存,只是虚拟地址 space

committed=197904KB

那是 197MB,不是 1GB

因此您没有显示 java 实际上为 class 数据消耗了 1GB 的内存,只是它保留了 1GB 的地址 space.

Java 为 类 保留 1G 的原因隐藏在它管理压缩 class 指针的方式中。

长答案:阅读此文档https://docs.oracle.com/javase/8/docs/technotes/guides/vm/gctuning/considerations.html

简短回答:在 'CompressedClassSpaceSize' 属性 -XX:CompressedClassSpaceSize=300m 中设置正确的大小