• TLAB


    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

  • 相关阅读:
    Instruments之Core Animation学习
    Instruments之Allocations
    Instruments之Activity Monitor使用入门
    Instruments之相关介绍(一)
    快速理解Java中的五种单例模式
    iOS单例详解
    eclipse设置代码自动提示
    iOS-静态库,动态库,framework,bundle浅析(四)
    8.0docker的客户端和守护进程
    1.0 docker介绍
  • 原文地址:https://www.cnblogs.com/mutong1228/p/10294706.html
Copyright © 2020-2023  润新知