java 中ThreadLocal实例分析
发布时间 - 2026-01-11 01:53:22 点击率:次java 中ThreadLocal实例分析

从概念上理解,threadlocal使变量在多个线程中相互隔离实现线程安全,threadlocal包装的变量最终都专属于对应的每个线程,线程之间相互独立,用一个具体实现来说明:
public interface Consumer {
int consume();
}
public class ComsumeThread implements Runnable {
private Consumer consumer;
public ComsumeThread(Consumer consumer) {
this.consumer = consumer;
}
@Override
public void run() {
for(int i=0;i<10;i++){
System.out.println(Thread.currentThread().getName()+" After Consume left:"+consumer.consume());
}
}
}
public class ConsumeClientA implements Consumer {
private static int leftNum = 30;
@Override
public int consume() {
int orgLeftNum = leftNum;
Random random = new Random(System.currentTimeMillis());
try {
Thread.sleep(random.nextInt(3));
} catch (InterruptedException e) {
e.printStackTrace();
}
orgLeftNum = orgLeftNum -1;
leftNum = orgLeftNum;
return leftNum;
}
public static void main(String[] args){
Consumer consumer = new ConsumeClientA();
Thread thread1 = new Thread(new ComsumeThread(consumer));
Thread thread2 = new Thread(new ComsumeThread(consumer));
Thread thread3 = new Thread(new ComsumeThread(consumer));
thread1.start();
thread2.start();
thread3.start();
}
}
ConsumeClientA是在没有做任何线程安全处理,结果如下:
Thread-2 After Consume left:29 Thread-1 After Consume left:29 Thread-3 After Consume left:29 Thread-2 After Consume left:28 Thread-1 After Consume left:28 Thread-3 After Consume left:28 Thread-2 After Consume left:27 Thread-1 After Consume left:27 Thread-2 After Consume left:26 Thread-3 After Consume left:27 Thread-1 After Consume left:25 Thread-2 After Consume left:25 Thread-3 After Consume left:25 Thread-1 After Consume left:24 Thread-2 After Consume left:24 Thread-3 After Consume left:24 Thread-1 After Consume left:23 Thread-2 After Consume left:23 Thread-3 After Consume left:23 Thread-1 After Consume left:22 Thread-2 After Consume left:22 Thread-3 After Consume left:22 Thread-1 After Consume left:21 Thread-2 After Consume left:21 Thread-3 After Consume left:21 Thread-1 After Consume left:20 Thread-2 After Consume left:20 Thread-3 After Consume left:20 Thread-1 After Consume left:19 Thread-3 After Consume left:18
增加threadlocal处理,每个线程相互独立,实现如下:
public class ConsumeClientB implements Consumer {
private ThreadLocal<Integer> leftNumThreadLocal = new ThreadLocal<Integer>(){
@Override
protected Integer initialValue() {
return 30;
}
};
@Override
public int consume() {
int orgLeftNum = leftNumThreadLocal.get();
Random random = new Random(System.currentTimeMillis());
try {
Thread.sleep(random.nextInt(3));
} catch (InterruptedException e) {
e.printStackTrace();
}
orgLeftNum = orgLeftNum -1;
leftNumThreadLocal.set(orgLeftNum);
return leftNumThreadLocal.get();
}
public static void main(String[] args){
Consumer consumer = new ConsumeClientB();
Thread thread1 = new Thread(new ComsumeThread(consumer));
Thread thread2 = new Thread(new ComsumeThread(consumer));
Thread thread3 = new Thread(new ComsumeThread(consumer));
thread1.start();
thread2.start();
thread3.start();
}
}
运行的结果如下:
Thread-1 After Consume left:29 Thread-3 After Consume left:29 Thread-2 After Consume left:29 Thread-1 After Consume left:28 Thread-3 After Consume left:28 Thread-2 After Consume left:28 Thread-1 After Consume left:27 Thread-3 After Consume left:27 Thread-2 After Consume left:27 Thread-1 After Consume left:26 Thread-3 After Consume left:26 Thread-2 After Consume left:26 Thread-1 After Consume left:25 Thread-3 After Consume left:25 Thread-2 After Consume left:25 Thread-1 After Consume left:24 Thread-3 After Consume left:24 Thread-2 After Consume left:24 Thread-1 After Consume left:23 Thread-3 After Consume left:23 Thread-2 After Consume left:23 Thread-1 After Consume left:22 Thread-3 After Consume left:22 Thread-2 After Consume left:22 Thread-1 After Consume left:21 Thread-3 After Consume left:21 Thread-2 After Consume left:21 Thread-1 After Consume left:20 Thread-3 After Consume left:20 Thread-2 After Consume left:20
每个线程拥有自己的独立变量,相互隔离实现线程安全。
那ThreadLocal是怎样实现这种线程隔离的线程安全的呢?
从ThreadLocal源码可以看到,真正实现线程隔离,与线程挂钩的,其实是ThreadLocal.ThreadLocalMap这个实现类,最明显的体现就在于Thread类源码的这样一个变量申明说明了ThreadLocal.ThreadLocalMap与Thread的关系:
ThreadLocal.ThreadLocalMap threadLocals, inheritableThreadLocals;
Thread类是包含threadLocals对象的,ThreadLocal的具体实现就是根据提供的get,set等接口,对当前thread的threadLocals变量进行相关操作的,如get操作代码如下:
public T get() {
Thread t = Thread.currentThread();
ThreadLocalMap map = getMap(t);
if (map != null) {
ThreadLocalMap.Entry e = map.getEntry(this);
if (e != null)
return (T)e.value;
}
return setInitialValue();
}
ThreadLocal.ThreadLocalMap getMap(Thread t) {
return t.threadLocals;
}
可以看到,getMap()方法就是从当前thread获取对应的threadLocals变量,然后从这个ThreadLocal.ThreadLocalMap类型的threadLocals变量中获取对应线程中该ThreadLocal对象对应的变量值。
set方法的操作也是一样:
public void set(T value) {
Thread t = Thread.currentThread();
ThreadLocal.ThreadLocalMap map = getMap(t);
if(map != null) {
map.set(this, value);
} else {
this.createMap(t, value);
}
}
void createMap(Thread t, T firstValue) {
t.threadLocals = new ThreadLocalMap(this, firstValue);
}
static class Entry extends WeakReference<ThreadLocal> {
Object value;
Entry(ThreadLocal var1, Object var2) {
super(var1);
this.value = var2;
}
}
ThreadLocalMap中存的是内部类Entry的数组,Entry是继承WeakReference实现,WeakReference的好处是保存对象引用,而又不干扰该对象被GC回收,线程执行完回收threadLocals变量时不会受到Entry封装的变量的干扰。
而且ThreadLocalMap中的key是ThreadLocal,所以一个ThreadLocal对象只能在一个Thread对象中保存一个ThreadLocal的value。
综上,很多人说ThreadLocal的实现是ThreadLocalMap中存Thread对象为key,变量为value的map结构,其实是错误的。
感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!
# java
# 中ThreadLocal
# ThreadLocal详解及实例
# java ThreadLocal使用案例详解
# java多线程编程之InheritableThreadLocal
# java 中ThreadLocal 的正确用法
# Java 并发编程之ThreadLocal详解及实例
# Java ThreadLocal 线程安全问题解决方案
# 实例详解Java中ThreadLocal内存泄露
# Java ThreadLocal的设计理念与作用
# 可以看到
# 自己的
# 的是
# 是在
# 多个
# 是从
# 希望能
# 是怎样
# 这样一个
# 人说
# 又不
# 谢谢大家
# 说明了
# 就在于
# 量为
# 概念上
# 综上
# 变量值
# 只能在
# Consume
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
laravel怎么配置Redis作为缓存驱动_laravel Redis缓存配置教程
高性价比服务器租赁——企业级配置与24小时运维服务
如何在服务器上三步完成建站并提升流量?
Laravel如何与Pusher实现实时通信?(WebSocket示例)
Laravel队列由Redis驱动怎么配置_Laravel Redis队列使用教程
Laravel如何实现全文搜索功能?(Scout和Algolia示例)
C++用Dijkstra(迪杰斯特拉)算法求最短路径
Laravel Telescope怎么调试_使用Laravel Telescope进行应用监控与调试
Laravel如何使用集合(Collections)进行数据处理_Laravel Collection常用方法与技巧
车管所网站制作流程,交警当场开简易程序处罚决定书,在交警网站查询不到怎么办?
Laravel如何自定义分页视图?(Pagination示例)
Laravel怎么进行数据库回滚_Laravel Migration数据库版本控制与回滚操作
Win11应用商店下载慢怎么办 Win11更改DNS提速下载【修复】
如何在IIS中新建站点并配置端口与IP地址?
Laravel Eloquent模型如何创建_Laravel ORM基础之Model创建与使用教程
通义万相免费版怎么用_通义万相免费版使用方法详细指南【教程】
Firefox Developer Edition开发者版本入口
Laravel怎么实现搜索功能_Laravel使用Eloquent实现模糊查询与多条件搜索【实例】
敲碗10年!Mac系列传将迎来「触控与联网」双革新
Laravel如何使用查询构建器?(Query Builder高级用法)
Laravel的路由模型绑定怎么用_Laravel Route Model Binding简化控制器逻辑
Mybatis 中的insertOrUpdate操作
Laravel怎么配置S3云存储驱动_Laravel集成阿里云OSS或AWS S3存储桶【教程】
Laravel项目怎么部署到Linux_Laravel Nginx配置详解
Laravel集合Collection怎么用_Laravel集合常用函数详解
如何快速搭建支持数据库操作的智能建站平台?
北京网页设计制作网站有哪些,继续教育自动播放怎么设置?
如何在橙子建站中快速调整背景颜色?
Laravel如何监控和管理失败的队列任务_Laravel失败任务处理与监控
阿里云网站搭建费用解析:服务器价格与建站成本优化指南
Python自然语言搜索引擎项目教程_倒排索引查询优化案例
javascript中闭包概念与用法深入理解
如何在云指建站中生成FTP站点?
Javascript中的事件循环是如何工作的_如何利用Javascript事件循环优化异步代码?
Windows11怎样设置电源计划_Windows11电源计划调整攻略【指南】
焦点电影公司作品,电影焦点结局是什么?
JS中使用new Date(str)创建时间对象不兼容firefox和ie的解决方法(两种)
Laravel Seeder怎么填充数据_Laravel数据库填充器的使用方法与技巧
Laravel如何编写单元测试和功能测试?(PHPUnit示例)
如何快速辨别茅台真假?关键步骤解析
LinuxShell函数封装方法_脚本复用设计思路【教程】
Laravel如何使用Spatie Media Library_Laravel图片上传管理与缩略图生成【步骤】
如何快速搭建高效简练网站?
标题:Vue + Vuex + JWT 身份认证的正确实践与常见误区解析
Laravel如何创建自定义Facades?(详细步骤)
如何用景安虚拟主机手机版绑定域名建站?
如何用5美元大硬盘VPS安全高效搭建个人网站?
安克发布新款氮化镓充电宝:体积缩小 30%,支持 200W 输出
如何在Windows 2008云服务器安全搭建网站?
Laravel怎么做数据加密_Laravel内置Crypt门面的加密与解密功能

