TLAB全称是Thread Local Allocation Buffer 即线程本地分配缓存,从名字上看是一个线程专用的内存分配区域,是为了加速对象分配而生的。
每一个线程都会产生一个TLAB,该线程独享的工作区域,java虚拟机使用这种TLAB区来避免多线程冲突问题,提高了
对象分配的效率。TLAB空间一般不会太大,当大对象无法再TLAB分配时,则会直接分配到堆上。
-XX:+UseTLAB 使用TLAB
-XX:+TLABSize 设置TLAB大小
-XX:TLABRefillWasteFraction设置维护进入TLAB空间的单个对象大小,他是一个比例值,默认为64,即如果对象大于整个空间的1/64,
则在堆创建对象
-XX:+PrintTLAB 查看TLAB信息
-XX:ResizeTLAB 自调整TLABRefillWasteFraction阈值。
实例一
public class TlabDemo { //参数: -Xmx30m -Xms30m -XX:+UseSerialGC -XX:+PrintGCDetails -XX:PretenureSizeThreshold=800 public static void main(String[] args){ Map<Integer,byte[]> map = new HashMap<Integer, byte[]>(); for(int i=0;i<5*1024;i++){ byte[] by = new byte[1024]; map.put(i,by); } } }
参数的含义是 堆初始大小 堆最大值 使用GC回收 打印GC信息 本地线程缓存大小阈值
如果新产生的对象大于800k,直接进入老年代,不在线程本地缓存中分配
[GC (Allocation Failure) [DefNew: 8192K->1024K(9216K), 0.0127060 secs] 9171K->6419K(29696K), 0.0128091 secs] [Times: user=0.02 sys=0.00, real=0.01 secs] Heap def new generation total 9216K, used 2424K [0x00000000fe200000, 0x00000000fec00000, 0x00000000fec00000) eden space 8192K, 17% used [0x00000000fe200000, 0x00000000fe35e060, 0x00000000fea00000) from space 1024K, 100% used [0x00000000feb00000, 0x00000000fec00000, 0x00000000fec00000) to space 1024K, 0% used [0x00000000fea00000, 0x00000000fea00000, 0x00000000feb00000) tenured generation total 20480K, used 5395K [0x00000000fec00000, 0x0000000100000000, 0x0000000100000000) the space 20480K, 26% used [0x00000000fec00000, 0x00000000ff144f90, 0x00000000ff145000, 0x0000000100000000) Metaspace used 3482K, capacity 4496K, committed 4864K, reserved 1056768K class space used 376K, capacity 388K, committed 512K, reserved 1048576K
打印信息说明:新生代和老年代加起来差不多就是30m,老年代5m,新生代的from space占比100%什么鬼意思?
实例二
//禁用TLAB //参数: -Xmx30m -Xms30m -XX:+UseSerialGC -XX:+PrintGCDetails -XX:PretenureSizeThreshold=800 -XX:-UseTLAB public static void main(String[] args){ Map<Integer,byte[]> map = new HashMap<Integer, byte[]>(); for(int i=0;i<5*1024;i++){ byte[] by = new byte[1024]; map.put(i,by); } }
Heap def new generation total 9216K, used 2275K [0x00000000fe200000, 0x00000000fec00000, 0x00000000fec00000) eden space 8192K, 27% used [0x00000000fe200000, 0x00000000fe438f28, 0x00000000fea00000) from space 1024K, 0% used [0x00000000fea00000, 0x00000000fea00000, 0x00000000feb00000) to space 1024K, 0% used [0x00000000feb00000, 0x00000000feb00000, 0x00000000fec00000) tenured generation total 20480K, used 7883K [0x00000000fec00000, 0x0000000100000000, 0x0000000100000000) the space 20480K, 38% used [0x00000000fec00000, 0x00000000ff3b2d20, 0x00000000ff3b2e00, 0x0000000100000000) Metaspace used 3481K, capacity 4496K, committed 4864K, reserved 1056768K class space used 376K, capacity 388K, committed 512K, reserved 1048576K
禁用TLAB之后,老年代的内存使用率明显上升