Netty源码分析-InternalThreadLocalMap_征服.刘华强的博客-程序员宅基地

技术标签: Netty  

 


class UnpaddedInternalThreadLocalMap {

    //ThreadLocal内部封装slowThreadLocalMap
    static final ThreadLocal<InternalThreadLocalMap> slowThreadLocalMap = new ThreadLocal<InternalThreadLocalMap>();

    //计数器
    static final AtomicInteger nextIndex = new AtomicInteger();

    /** Used by {@link FastThreadLocal} */
    Object[] indexedVariables;

    //成员变量
    // Core thread-locals
    int futureListenerStackDepth;
    int localChannelReaderStackDepth;
    Map<Class<?>, Boolean> handlerSharableCache;
    IntegerHolder counterHashCode;
    ThreadLocalRandom random;
    Map<Class<?>, TypeParameterMatcher> typeParameterMatcherGetCache;
    Map<Class<?>, Map<String, TypeParameterMatcher>> typeParameterMatcherFindCache;

    // String-related thread-locals
    StringBuilder stringBuilder;
    Map<Charset, CharsetEncoder> charsetEncoderCache;
    Map<Charset, CharsetDecoder> charsetDecoderCache;

    // ArrayList-related thread-locals
    ArrayList<Object> arrayList;

    UnpaddedInternalThreadLocalMap(Object[] indexedVariables) {
        this.indexedVariables = indexedVariables;
    }
}

 


//集成了UnpaddedInternalThreadLocalMap
//内部包含了一个静态的threadlocal类slowThreadLocalMap和一些数据结构
public final class InternalThreadLocalMap extends UnpaddedInternalThreadLocalMap {

    private static final InternalLogger logger = InternalLoggerFactory.getInstance(InternalThreadLocalMap.class);

    private static final int DEFAULT_ARRAY_LIST_INITIAL_CAPACITY = 8;
    private static final int STRING_BUILDER_INITIAL_SIZE;
    private static final int STRING_BUILDER_MAX_SIZE;

    public static final Object UNSET = new Object();

    private BitSet cleanerFlags;

    static {
        STRING_BUILDER_INITIAL_SIZE =
                SystemPropertyUtil.getInt("io.netty.threadLocalMap.stringBuilder.initialSize", 1024);
        logger.debug("-Dio.netty.threadLocalMap.stringBuilder.initialSize: {}", STRING_BUILDER_INITIAL_SIZE);

        STRING_BUILDER_MAX_SIZE = SystemPropertyUtil.getInt("io.netty.threadLocalMap.stringBuilder.maxSize", 1024 * 4);
        logger.debug("-Dio.netty.threadLocalMap.stringBuilder.maxSize: {}", STRING_BUILDER_MAX_SIZE);
    }

    //如果设置了就获取否则返回null
    public static InternalThreadLocalMap getIfSet() {
        //拿到当前线程
        Thread thread = Thread.currentThread();
        //如果是FastThreadLocalThread类型,则直接获取InternalThreadLocalMap对象
        if (thread instanceof FastThreadLocalThread) {
            return ((FastThreadLocalThread) thread).threadLocalMap();
        }
        //否则上ThreadLocal获取
        return slowThreadLocalMap.get();
    }

    //获取InternalThreadLocalMap对象,没有就创建一个
    public static InternalThreadLocalMap get() {
        Thread thread = Thread.currentThread();
        if (thread instanceof FastThreadLocalThread) {
            return fastGet((FastThreadLocalThread) thread);
        } else {
            return slowGet();
        }
    }

    private static InternalThreadLocalMap fastGet(FastThreadLocalThread thread) {
        //获取线程的成员变量threadLocalMap();
        InternalThreadLocalMap threadLocalMap = thread.threadLocalMap();
        //如果为空则创建一个新的设置到线程当中
        if (threadLocalMap == null) {
            thread.setThreadLocalMap(threadLocalMap = new InternalThreadLocalMap());
        }
        return threadLocalMap;
    }

    private static InternalThreadLocalMap slowGet() {
        //在ThreadLocal当中获取InternalThreadLocalMap对象
        ThreadLocal<InternalThreadLocalMap> slowThreadLocalMap = UnpaddedInternalThreadLocalMap.slowThreadLocalMap;
        InternalThreadLocalMap ret = slowThreadLocalMap.get();
        //如果为null则创建一个新的设置到ThreadLocal当中并返回
        if (ret == null) {
            ret = new InternalThreadLocalMap();
            slowThreadLocalMap.set(ret);
        }
        return ret;
    }

    //移除InternalThreadLocalMap
    public static void remove() {
        //获取当前线程
        Thread thread = Thread.currentThread();
        //如果是FastThreadLocalThread类型则直接把成员变量设置为null
        if (thread instanceof FastThreadLocalThread) {
            ((FastThreadLocalThread) thread).setThreadLocalMap(null);
        } else {
            //在ThreadLocal当中移除
            slowThreadLocalMap.remove();
        }
    }

    //在ThreadLocal当中移除
    public static void destroy() {
        slowThreadLocalMap.remove();
    }

    //计数器++
    public static int nextVariableIndex() {
        int index = nextIndex.getAndIncrement();
        //如果为负数说明已经++到Int的最大值
        if (index < 0) {
            //计数器-- 后抛出异常
            nextIndex.decrementAndGet();
            throw new IllegalStateException("too many thread-local indexed variables");
        }
        return index;
    }

    //获取最后一个变量的index
    public static int lastVariableIndex() {
        return nextIndex.get() - 1;
    }

    // Cache line padding (must be public)
    // With CompressedOops enabled, an instance of this class should occupy at least 128 bytes.
    public long rp1, rp2, rp3, rp4, rp5, rp6, rp7, rp8, rp9;

    //构造参数 传入32个长度的数组里面全部都是相同的object引用
    private InternalThreadLocalMap() {
        super(newIndexedVariableTable());
    }

    //创建一个32长队数组,里面全部引用一个相同的object对象
    private static Object[] newIndexedVariableTable() {
        Object[] array = new Object[32];
        Arrays.fill(array, UNSET);
        return array;
    }

    //统计数量
    public int size() {
        int count = 0;

        if (futureListenerStackDepth != 0) {
            count ++;
        }
        if (localChannelReaderStackDepth != 0) {
            count ++;
        }
        if (handlerSharableCache != null) {
            count ++;
        }
        if (counterHashCode != null) {
            count ++;
        }
        if (random != null) {
            count ++;
        }
        if (typeParameterMatcherGetCache != null) {
            count ++;
        }
        if (typeParameterMatcherFindCache != null) {
            count ++;
        }
        if (stringBuilder != null) {
            count ++;
        }
        if (charsetEncoderCache != null) {
            count ++;
        }
        if (charsetDecoderCache != null) {
            count ++;
        }
        if (arrayList != null) {
            count ++;
        }

        for (Object o: indexedVariables) {
            if (o != UNSET) {
                count ++;
            }
        }

        // We should subtract 1 from the count because the first element in 'indexedVariables' is reserved
        // by 'FastThreadLocal' to keep the list of 'FastThreadLocal's to remove on 'FastThreadLocal.removeAll()'.
        return count - 1;
    }

    //返回StringBuilder
    public StringBuilder stringBuilder() {
        StringBuilder sb = stringBuilder;
        if (sb == null) {
            //没有则新创建一个返回 STRING_BUILDER_INITIAL_SIZE=初始容量
            return stringBuilder = new StringBuilder(STRING_BUILDER_INITIAL_SIZE);
        }

        //如果之前存在sb, 则判断容量是否 大于 STRING_BUILDER_MAX_SIZE
        if (sb.capacity() > STRING_BUILDER_MAX_SIZE) {

            //长度设置为初始容量
            sb.setLength(STRING_BUILDER_INITIAL_SIZE);
            sb.trimToSize();
        }
        //把长度设置为0,相当于清空sb
        sb.setLength(0);
        return sb;
    }


    //返回Map<Charset, CharsetEncoder>没有则创建
    public Map<Charset, CharsetEncoder> charsetEncoderCache() {
        Map<Charset, CharsetEncoder> cache = charsetEncoderCache;
        if (cache == null) {
            charsetEncoderCache = cache = new IdentityHashMap<Charset, CharsetEncoder>();
        }
        return cache;
    }

    //返回Map<Charset, CharsetDecoder>没有则创建
    public Map<Charset, CharsetDecoder> charsetDecoderCache() {
        Map<Charset, CharsetDecoder> cache = charsetDecoderCache;
        if (cache == null) {
            charsetDecoderCache = cache = new IdentityHashMap<Charset, CharsetDecoder>();
        }
        return cache;
    }

    public <E> ArrayList<E> arrayList() {
        return arrayList(DEFAULT_ARRAY_LIST_INITIAL_CAPACITY);
    }

   //返回一个minCapacity大小的list
    @SuppressWarnings("unchecked")
    public <E> ArrayList<E> arrayList(int minCapacity) {
        ArrayList<E> list = (ArrayList<E>) arrayList;
        if (list == null) {
            arrayList = new ArrayList<Object>(minCapacity);
            return (ArrayList<E>) arrayList;
        }
        list.clear();
        list.ensureCapacity(minCapacity);
        return list;
    }

    public int futureListenerStackDepth() {
        return futureListenerStackDepth;
    }

    public void setFutureListenerStackDepth(int futureListenerStackDepth) {
        this.futureListenerStackDepth = futureListenerStackDepth;
    }

    //获得线程的Random对象
    public ThreadLocalRandom random() {
        ThreadLocalRandom r = random;
        if (r == null) {
            random = r = new ThreadLocalRandom();
        }
        return r;
    }

    public Map<Class<?>, TypeParameterMatcher> typeParameterMatcherGetCache() {
        Map<Class<?>, TypeParameterMatcher> cache = typeParameterMatcherGetCache;
        if (cache == null) {
            typeParameterMatcherGetCache = cache = new IdentityHashMap<Class<?>, TypeParameterMatcher>();
        }
        return cache;
    }

    public Map<Class<?>, Map<String, TypeParameterMatcher>> typeParameterMatcherFindCache() {
        Map<Class<?>, Map<String, TypeParameterMatcher>> cache = typeParameterMatcherFindCache;
        if (cache == null) {
            typeParameterMatcherFindCache = cache = new IdentityHashMap<Class<?>, Map<String, TypeParameterMatcher>>();
        }
        return cache;
    }

    public IntegerHolder counterHashCode() {
        return counterHashCode;
    }

    public void setCounterHashCode(IntegerHolder counterHashCode) {
        this.counterHashCode = counterHashCode;
    }

    public Map<Class<?>, Boolean> handlerSharableCache() {
        Map<Class<?>, Boolean> cache = handlerSharableCache;
        if (cache == null) {
            // Start with small capacity to keep memory overhead as low as possible.
            handlerSharableCache = cache = new WeakHashMap<Class<?>, Boolean>(4);
        }
        return cache;
    }

    public int localChannelReaderStackDepth() {
        return localChannelReaderStackDepth;
    }

    public void setLocalChannelReaderStackDepth(int localChannelReaderStackDepth) {
        this.localChannelReaderStackDepth = localChannelReaderStackDepth;
    }

    public Object indexedVariable(int index) {
        Object[] lookup = indexedVariables;
        return index < lookup.length? lookup[index] : UNSET;
    }

    /**
     * @return {@code true} if and only if a new thread-local variable has been created
     */
    //存储对象
    public boolean setIndexedVariable(int index, Object value) {
        //对象数组
        Object[] lookup = indexedVariables;
        //判断下标小于数组长度
        if (index < lookup.length) {
            //拿出旧值
            Object oldValue = lookup[index];
            //设置新值
            lookup[index] = value;
            //判断oldValue == UNSET第一次设置
            return oldValue == UNSET;
        } else {
            //下标大于数组长度需要扩容
            expandIndexedVariableTableAndSet(index, value);
            return true;
        }
    }

    private void expandIndexedVariableTableAndSet(int index, Object value) {
        Object[] oldArray = indexedVariables;
        //旧数组长度
        final int oldCapacity = oldArray.length;
        //根据index扩容
        int newCapacity = index;
        newCapacity |= newCapacity >>>  1;
        newCapacity |= newCapacity >>>  2;
        newCapacity |= newCapacity >>>  4;
        newCapacity |= newCapacity >>>  8;
        newCapacity |= newCapacity >>> 16;
        newCapacity ++;

        //把旧数组拷贝到新数组
        Object[] newArray = Arrays.copyOf(oldArray, newCapacity);
        //其余位置全部设置为UNSET
        Arrays.fill(newArray, oldCapacity, newArray.length, UNSET);
        //设置值
        newArray[index] = value;
        //更新成员变量
        indexedVariables = newArray;
    }

    //移除对象
    public Object removeIndexedVariable(int index) {
        Object[] lookup = indexedVariables;
        if (index < lookup.length) {
            Object v = lookup[index];
            lookup[index] = UNSET;
            return v;
        } else {
            return UNSET;
        }
    }

    public boolean isIndexedVariableSet(int index) {
        Object[] lookup = indexedVariables;
        return index < lookup.length && lookup[index] != UNSET;
    }

    public boolean isCleanerFlagSet(int index) {
        return cleanerFlags != null && cleanerFlags.get(index);
    }

    public void setCleanerFlag(int index) {
        if (cleanerFlags == null) {
            cleanerFlags = new BitSet();
        }
        cleanerFlags.set(index);
    }
}

 

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/nimasike/article/details/91432574

智能推荐

使用ssh登录shell脚本远程调用_ssh脚本_神奇时代的博客-程序员宅基地

环境:两台centos 7.0 ip:192.168.1.228、192.168.1.229从192.168.1.228免输入密码登录到192.168.1.229配置192.168.1.2281、进入~目录cd ~创建.ssh 文件夹mkdir .sshchmod 700 .ssh2、创建ssh钥匙文件ssh-keygen -t  rsa3、更改私有

计算机主机的拆卸步骤,电脑主板怎么拆CPU 主板拆CPU步骤图文教程_weixin_39745345的博客-程序员宅基地

对于很多朋友来说,拆电脑主机内部零件觉得并不是什么难事,但拆主板上的CPU并非易事,搞不好还可能损伤CPU和插槽,CPU与插槽、CPU风扇紧密相连,所以你懂的。每台CPU和CPU风扇的安装不一定相同,但大同小异。下面详细介绍电脑主板怎么拆CPU的方法。主板拆CPU步骤:1、首先打开电脑主机,我们先来认识下,主板CPU上有CPU风扇,风扇还会有电源线连接供电。我们就先拔开风扇电源线。2、接着钮转风扇...

Json - Java_yuxiaohui78的博客-程序员宅基地

从json lib网站下载json java支持包http://json-lib.sourceforge.net/index.html同时需要下载依赖包:GroupIdArtifactIdVersionClassifierTypeOptionalcommons-beanutilscommons-bean

springboot的5种读取配置方式(5):通过applicationContext.xml读取_灰太狼_cxh的博客-程序员宅基地

5.通过application.xml读取:/** * 学生实体类 * Created by ASUS on 2018/5/4 */@Component(&quot;Student&quot;)public class Student { private String name; private int age; public String getName() { ...

算法与数据结构——零、目录_桥苯环萘我老婆的博客-程序员宅基地

目录前oier最后的倔强。看着名词都是会过的,至少曾经会用来装逼啊!感谢王争大佬在课程上的总结之后的总结可能是自己整合,不会与专栏利益相关数据结构部分一、线性表1、数组2、链表* 单链表 * 双向链表 * 循环链表 * 双向循环链表 * 静态链表3、栈* 顺序栈 * 链式栈4、队列* 普通队列 * 双端队列 * 阻塞队列...

shell if_金士顿的博客-程序员宅基地

 if 语句语法格式:if conditionthen command1 command2 ... commandN fi写成一行(适用于终端命令提示符):if [ $(ps -ef | grep -c &quot;ssh&quot;) -gt 1 ]; then echo &quot;true&quot;; fi末尾的fi就是if倒过来拼写,后面还会遇到类似的。if e...

随便推点

部署Harbor v1.8.1 开启helm-charts_chaixia4264的博客-程序员宅基地

1、部署前准备下载Harborv1.8.1下载在线包wget https://storage.googleapis.com/harbor-releases/release-1.8.0/harbor-online-installer-v1.8.1.tgz由于文件使用的谷歌云,可能要使用特殊的手段才能下载下面是几个文件内容2、修改配置文件主要修改地方hostname: docker.t...

同步非阻塞IO例子_晏无心_新浪博客_wangchao_yun的博客-程序员宅基地

同步非阻塞IO是指在进行IO读写时,可以立即返回,不用阻塞等待read、write完全结束。java实现同步非阻塞IO需要使用JavaNIO技术。先来看server端:使用iavanio的ServerSocketChannel,设置serverChannel.configureBlocking(true);是为了方便测试,serverSocket阻塞的等待客户端的连接,连...

Android 4.4系统HTTPS证书校验_安卓4.4https_梦翼-的博客-程序员宅基地

Android系统在建立HTTPS三次握手后会进行证书校验,接下来根据android 7.0系统代码来看一下证书校验流程 从org.conscrypt.TrustManagerImpl开始介绍1、证书校验入口函数  @Override public void checkServerTrusted(X509Certificate[] chain, String authType)

链路两段不同网段怎么通信_静态路由跨网段访问配置方法和命令_索嵩的博客-程序员宅基地

静态路由跨网段访问配置方法和命令静态路由是一种需要管理员手工配置的特殊路由。静态路由比动态路由使用更少的带宽,并且不占用CPU资源来计算和分析路由更新。但是当网络发生故障或者拓扑发生变化后,静态路由不会自动更新,必须手动重新配置。静态路由有5个主要的参数:目的地址和掩码、出接口和下一跳、优先级。使用静态路由的好处是配置简单、可控性高,当网络结构比较简单时,只需配置静态路由就可以使网络正常工作。在复...

关于Web实现前后端分离,前后端解耦_weixin_30565327的博客-程序员宅基地

一、前言”前后端分离“已经成为互联网项目开发的业界标杆,通过Tomcat+Ngnix(也可以中间有个Node.js),有效地进行解耦。并且前后端分离会为以后的大型分布式架构、弹性计算架构、微服务架构、多端化服务(多种客户端,例如:浏览器,车载终端,安卓,IOS等等)打下坚实的基础。前后端分离(解耦)的核心思想是:前端Html页面通过Ajax调用后端的RestFul API并使用Jso...

交换机ssh版本信息可被获取_迈威多光口三层万兆嵌入式工业交换机,助力智慧矿山数字化建设..._超高校级的社畜的博客-程序员宅基地

随着矿井生产规模的逐步扩大及系统的不断完善,井下原有的千兆环网系统已不能满足矿井发展需求,随着煤矿开采的规模不断扩大,矿井下接入的系统越来越多,特别是高清摄像仪的数量不断增多,信息传输量出现爆发式增长。在一些大型煤矿,千兆以太环网已无法满足井下信息传输带宽的要求,万兆以太环网成为必然趋势。MISCOM8112GX-4XGF-8GC是迈威通信根据行业实际需求研发设计的一款三层万兆嵌入式环网工业以太网...