ThreadLocal的说明

为了线程间内的变量能够共享,但又不与其它线程共享,我们会使用ThreadLocal。ThreadLocal是与线程绑定的,可以理解为一个key是线程对象,value是变量的一个map。但在使用ThreadLocal时需要注意的是,如果Thread在A用户访问完后,被B用户继续使用那么A用户存的变量会被B用户访问到。这根具体的容器有关。比如jetty就没问题,tomcat6.5就有问题。因为tomcat它会维护一个用户线程池,可能是考虑到性能以及资源上的考虑。即A用户访问是使用Thread1,那么当它访问结束后,Thread1并不是马上销毁,而是交还给线程池,那么B再来访问时可能就会使用到Thread1。而jetty它的机制可能每次都是使用新的,使用完后就销毁。(但如果有并发,即A未访问结束,此时会分配一个新的线程给B用户。) 因此使用ThreadLocal最好的做法是让清空变量,即ThreadLocal.set(null); 这里我们也可以看到其实线程我们可以理解为执行一个动作的载体。
以下是 `ThreadLocal` 类中 `set` 方法原理的图示说明以及对应的文字解释: #### 整体流程 ```plaintext +---------------------+ | ThreadLocal.set(T) | +---------------------+ | v +---------------------+ | 获取当前线程对象 | +---------------------+ | v +---------------------+ | 从线程对象获取 | | ThreadLocalMap | +---------------------+ | | 存在? | +----+----+ | | v v +-------+ +---------------------+ | 创建 | | 调用 ThreadLocalMap | | Map | | 的 set 方法 | +-------+ +---------------------+ ``` #### 详细步骤及代码对应 1. **获取当前线程对象** ```java Thread t = Thread.currentThread(); ``` 这一步是获取当前正在执行的线程对象。 2. **获取线程对应的 `ThreadLocalMap`** ```java ThreadLocalMap map = getMap(t); ``` 每个线程都有一个自己的 `ThreadLocalMap` 对象,这里通过当前线程对象获取对应的 `ThreadLocalMap`。 3. **判断 `ThreadLocalMap` 是否存在** ```java if (map != null) map.set(this, value); else createMap(t, value); ``` 如果 `ThreadLocalMap` 已经存在,则调用其 `set` 方法;如果不存在,则创建一个新的 `ThreadLocalMap`。 #### `ThreadLocalMap.set` 方法内部原理图示 ```plaintext +---------------------+ | ThreadLocalMap.set | +---------------------+ | v +---------------------+ | 计算数组下标 | | i = key.threadLocalHashCode & (len - 1) | +---------------------+ | v +---------------------+ | 检查数组位置 | | e = tab[i] | +---------------------+ | | e 是否为空? | +----+----+ | | v v +-------+ +---------------------+ | 插入 | | 检查 key 是否相同 | | Entry | | 是: 覆盖 value | | | | 否: 线性探测下一个 | | | | 位置 | +-------+ +---------------------+ | | 检查 key 是否为 null? | +----+----+ | | v v +-------+ +---------------------+ | 跳过 | | 替换过期 Entry | | | | 清理过期对象 | +-------+ +---------------------+ | | 插入新 Entry | v +---------------------+ | 检查是否需要扩容 | | sz >= threshold | | 是: rehash() | | 否: 结束 | +---------------------+ ``` #### 代码对应 ```java private void set(ThreadLocal<?> key, Object value) { Entry[] tab = table; int len = tab.length; int i = key.threadLocalHashCode & (len-1); for (Entry e = tab[i]; e != null; e = tab[i = nextIndex(i, len)]) { ThreadLocal<?> k = e.get(); if (k == key) { e.value = value; return; } if (k == null) { replaceStaleEntry(key, value, i); return; } } tab[i] = new Entry(key, value); int sz = ++size; if (!cleanSomeSlots(i, sz) && sz >= threshold) rehash(); } ``` ###
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值