域名

大话ThreadLocal

时间:2010-12-5 17:23:32  作者:数据库   来源:人工智能  查看:  评论:0
内容摘要:话说大唐贞观年间,物华天宝,人杰地灵。太宗治下国家一派祥和,四方来朝。这些各国前来朝见的使臣,也在暗中较量,比较进贡宝贝。一开始朝中只提供了一个存放贡品,各国使臣都可以进入。这样一来在贡品呈现时会发现

话说大唐贞观年间,大话物华天宝,大话人杰地灵。大话太宗治下国家一派祥和,大话四方来朝。大话

这些各国前来朝见的大话使臣,也在暗中较量,大话比较进贡宝贝。大话一开始朝中只提供了一个存放贡品,大话各国使臣都可以进入。大话这样一来在贡品呈现时会发现有些已经被偷偷调了包,大话甚至有些被偷走了。大话

大胆,大话竟然有人敢在太岁头上动土。大话太宗震怒,大话于是「贡品处」被重兵把守。而这里存的东西,有时各国使臣还会做些修饰工作,无形中加大了皇城守卫的工作量。守卫统领上书建议,将各国使臣的物品,都放在「贡品处」为他们自己分配的「小柜子」里。只有他们自己可以打开。

这时不需要人把守,也都井然有序。四方使者的东西也没正丢失弄错过。

大殿里歌舞表演ing,网站模板太宗高兴地看着各国送来的宝贝,擦了一把嘴边的油说:

这「小柜子」真是好呀。对于每年都来朝见的使者,这个柜子一直给他留着,每年来都用啊。

对于每年新来的使者,他们的柜子怎么办呢?放心好了,负责被褥发放的会根据当前使者存放物品提供一个等规格的柜子。

这里各国使臣就像我们多线程一样,都在向应用中非线程安全的一个地方写数据,因此很容易出现数据错乱、丢失等情况。

为了保证线程的执行安全,可以为方法进行加锁。但重兵把守后,所有来的请求都需要进行排队执行,效率上打了折扣。

而上面说的「小柜子」,就是我们本文的主角:ThreadLocal。云服务器提供商对于每个不同的使者,分配的是不同的柜子,这样他们之间的数据就被隔离开来,互不影响。

新的柜子分配就是 ThreadLocal对于一个新线程提供initValue的实现。

在多线程的应用环境中,为了多个线程间的数据互不影响,我们可以通过加锁,栈封闭等多种方式来实现, ThreadLocal也是一种。

ThreadLocal 这个类的名称起的很好,类如其名,local,相当于是一个线程的本地数据,这样每个线程的数据都存在自己的local里,互不影响,云服务器各自占山为王

也是逍遥自在。

回到代码,我们来看 ThreadLocal 是如何和各个 Thread 之间建立起关联的呢?

我们来看,每个Thread,都有这样一个属性,一个ThreadLocal.ThreadLocalMap的属性,能互不影响的秘密都在这里。

/* ThreadLocal values pertaining to this thread. This map is maintained      * by the ThreadLocal class. */     ThreadLocal.ThreadLocalMap threadLocals = null; 

这个ThreadLoalMap是什么时候被设置值的呢?

我们来看ThreadLocal的使用。

一般的用法是:

ThreadLocal<Integer> local = new ThreadLocal<Integer>() {  protected Integer initialValue() {                      return 1;                 } }; 

然后使用这个ThreadLocal变量进行set和get操作。

set的时候,会先判断对于当前线程,是否已经分配了map,没有则创建。

public void set(T value) {          Thread t = Thread.currentThread();         ThreadLocalMap map = getMap(t);         if (map != null)             map.set(this, value);         else             createMap(t, value);     } 

是否已经分配过map就是根据当前线程的 theThreadLocals 属性来判断的

ThreadLocalMap getMap(Thread t) {      return t.threadLocals; } 

那createMap的时候,就会给当前线程的threadLocals赋值

void createMap(Thread t, T firstValue) {      t.threadLocals = new ThreadLocalMap(this, firstValue); } 

这个ThreadLocalMap里是以数组的形式放的多个Entry。

在 get 的时候,如果没数据会根据上面的initValue方法创建一个新的返回。这样多个线程用的就是不同的东西了。

那这里还有一点,对于不同的东西, ThreadLocal 可以通过泛型做区分,当然你也能一股脑的放到一起,那取的时候就费劲了。

【本文为专栏作者“侯树成”的原创稿件,转载请通过作者微信公众号『Tomcat那些事儿』获取授权】

戳这里,看该作者更多好文

copyright © 2025 powered by 益强资讯全景  滇ICP备2023006006号-31sitemap