TLAB的全稱是Thread Local Allocation Buffer,即線程本地分配緩存區,這是一個線程專用的內存分配區域。
由於對象一般會分配在堆上,而堆是全局共享的。因此在同一時間,可能會有多個線程在堆上申請空間。
因此,每次對象分配都必須要進行同步(虛擬機采用CAS配上失敗重試的方式保證更新操作的原子性),而在競爭激烈的場合分配的效率又會進一步下降。
JVM使用TLAB來避免多線程沖突,在給對象分配內存時,每個線程使用自己的TLAB,這樣可以避免線程同步,提高了對象分配的效率。
-XX:+UseTLAB 使用TLAB
-XX:+TLABSize 設置TLAB大小
-XX:TLABRefillWasteFraction設置維護進入TLAB空間的單個對象大小,他是一個比例值,默認為64,即如果對象大於整個空間的1/64,
-XX:TLABWasteTargetPercent設置TLAB空間所占用Eden空間的百分比大小 默認是1%
則在堆創建對象
-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之后,老年代的內存使用率明顯上升
實例三
逃逸分析
public static void alloc(){ byte[] b = new byte[3]; } public static void main(String[] args){ //TLAB分配 //參數: -XX:+UseTLAB -XX:+PrintTLAB -XX:+PrintGC -XX:TLABSize=102400 -XX:-ResizeTLAB -XX:TLABRefillWasteFraction=100 -XX:-DoEscapeAnalysis -server for(int i=0;i<1000000;i++){ alloc(); }
}
在具體測試TLAB之前復習幾個概念
為對象分配空間的任務等同於把一個確定大小的內存從java堆中划分出來
常見的方法有指針碰撞和空閑列表
對象分配的流程
如果開啟棧上分配,JVM會先進行棧上分配,如果沒有開啟棧上分配或則不符合條件的則會進行TLAB分配,如果TLAB分配不成功,再嘗試在eden區分配,如果對象滿足了直接進入老年代的條件,那就直接分配在老年代。
后續學習的博客地址
源碼分析
https://www.jianshu.com/p/cd85098cca39
https://www.jianshu.com/p/2343f2c0ecc4