博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
threadlocal原理及常用应用场景
阅读量:5377 次
发布时间:2019-06-15

本文共 2571 字,大约阅读时间需要 8 分钟。

1.深入解析ThreadLocal类

ThreadLocal类提供的几个方法:

  1. public T get() { }  
  2. public void set(T value) { }  
  3. public void remove() { }  
  4. protected T initialValue() { }

  get()方法是用来获取ThreadLocal在当前线程中保存的变量副本,set()用来设置当前线程中变量的副本,remove()用来移除当前线程中变量的副本,initialValue()是一个protected方法,一般是用来在使用时进行重写的,它是一个延迟加载方法

  

  一个以ThreadLocal对象为键、任意对象为值的存储结构。
  是一个数据结构,有点像HashMap,可以保存"key : value"键值对,但是一个ThreadLocal只能保存一个,并且各个线程的数据互不干扰。
  该结构被附带在线程上,也就是说一个线程可以根据一个ThreadLocal对象查询到绑定在这个线程上的一个值。
 
  
ThreadLocal
local = new ThreadLocal();local.set("test");String demo = local.get();

在线程A中初始化了一个ThreadLocal对象local,并set了一个值test,同时在线程A中通过get可拿到之前设置的值,但是如果在线程B中,拿到的将是一个null

这是如何实现的呢?之前说过,ThreadLocal保证了各个线程的数据互不干扰

  看看set(T value)和get()方法的源码

  

    // 返回当前线程该线程局部变量副本中的值        public T get() {            Thread t = Thread.currentThread();            ThreadLocalMap map = getMap(t);            if (map != null) {                ThreadLocalMap.Entry e = map.getEntry(this);                if (e != null) {                    @SuppressWarnings("unchecked")                    T result = (T)e.value;                    return result;                }            }            return setInitialValue();        }       /**         *设置此线程局部变量的当前线程的副本到指定的值         *大多数的子类都不需要重写此方法         */        public void set(T value) {            Thread t = Thread.currentThread();            ThreadLocalMap map = getMap(t);            if (map != null)                map.set(this, value);            else                createMap(t, value);}   /**     * Get the map associated with a ThreadLocal. Overridden in     * InheritableThreadLocal.     *     * @param  t the current thread     * @return the map     */    ThreadLocalMap getMap(Thread t) {        return t.threadLocals;    }

 

可以发现,每个线程中都有一个ThreadLocalMap数据结构

执行set时,其值是保存在当前线程的threadLocals变量中,执行get时,从当前线程的threadLocals变量获取

所以在线程A中set的值,对线程B来说是摸不到的,而且在线程B中重新set的话,也不会影响到线程A中的值,保证了线程之间不会相互干扰

 

ThreadLocalMap是一个类似HashMap的数据结构,但是在ThreadLocal中,并没实现Map接口;

在ThreadLoalMap中,也是初始化一个大小为16的Entry数组table,Entry节点对象用来保存每一个key-value键值对,这里的key永远都是ThreadLocal对象,通过ThreadLocal对象的set方法,结果把ThreadLocal对象自己当做key;

ThreadLoalMap的Entry是继承WeakReference,和HashMap很大的区别是,Entry中没有next字段,所以不存在链表的情况;

 

 2.需要注意:

ThreadLocal 不是用来解决共享对象的多线程访问问题的,一般情况下,通过set() 到线程中的对象是该线程自己使用的对象,其他线程是不需要访问的,也访问不到的。各个线程中访问的是不同的对象。

另外,说ThreadLocal使得各线程能够保持各自独立的一个对象,并不是通过set()来实现的,而是通过每个线程中的new 对象 的操作来创建的对象,每个线程创建一个,不是什么对象的拷贝或副本。

通过set()将这个新创建的对象的引用保存到各线程的自己的一个map中,每个线程都有这样一个map,执行get()时,各线程从自己的map中取出放进去的对象,因此取出来的是各自自己线程中的对象,ThreadLocal实例是作为map的key来使用的。

 

 

转载于:https://www.cnblogs.com/xushuai0620/p/8242458.html

你可能感兴趣的文章
JavaScript总结之DOM基本操作(三)
查看>>
为Vmware硬盘减肥瘦身
查看>>
python-flask学习
查看>>
Controller与View数据传递 多Model传递
查看>>
arm 汇编小练习
查看>>
RegQueryValueEx函数
查看>>
漫谈数据库索引
查看>>
【NOIP2004】合唱队形
查看>>
spring面试题
查看>>
python使用pickle,json等序列化dict
查看>>
php进行文件的强制下载
查看>>
每日python(6)
查看>>
Python正则表达式中的re.S的作用
查看>>
ubuntu15.10运行android studio出错unable to run mksdcard sdk tool
查看>>
HashMap面试知多少
查看>>
Effective C# 学习笔记(二十七)使你的类型可被序列化
查看>>
LDAP客户端配置
查看>>
(转)NAT原理与NAT穿越
查看>>
13.内存原理
查看>>
24.函数信号机制(本质上就是函数指针)
查看>>