ThreadLocal使用详解-从源码层面分析

从demo入手看效果

代码Demo

    static ThreadLocal tl1 = new ThreadLocal();static ThreadLocal tl2 = new ThreadLocal();static ThreadLocal tl3 = new ThreadLocal();public static void main(String[] args) {tl1.set("123");tl2.set("456");tl3.set("4586");Thread t1 = new Thread(() -> {System.out.println("t1:tl1-before:" + tl1.get());System.out.println("t1:tl2-before:" + tl2.get());System.out.println("t1:tl3-before:" + tl3.get());ThreadLocal tl4 = new ThreadLocal();tl1.set("1231");tl2.set("4561");tl3.set("12312");tl4.set("123121231212312");System.out.println("t1:tl1-" + tl1.get());System.out.println("t1:tl2-" + tl2.get());System.out.println("t1:tl3-" + tl3.get());System.out.println("t1:tl4-" + tl4.get());});t1.start();Thread.sleep(5000);System.out.println("main:tl1-" + tl1.get());System.out.println("main:tl2-" + tl2.get());System.out.println("main:tl3-" + tl3.get());

执行结果:

t1:tl1-before:null
t1:tl2-before:null
t1:tl3-before:null
t1:tl1-1231
t1:tl2-4561
t1:tl3-12312
t1:tl4-123121231212312
main:tl1-123
main:tl2-456
main:tl3-4586

结论:
根据结果可以发现 相同的ThreadLocal 在main线程和t1 线程存入不同的数据,拿到的结果不一样,即使主线程先在ThreadLocal执行set方法,在t1线程中还是获取为空,可以说明在T1线程内部获取ThreadLocal是获取不到main线程的,因此可以证明实现了线程隔离。

源码分析

Thread:

 ThreadLocal.ThreadLocalMap threadLocals = null;

ThreadLocal类中包含一个静态类 ThreadLocalMap,是一个定制化的哈希表,用于存储线程的私有数据。键为ThreadLocal对象(弱引用),值为线程的局部变量(强引用)
在这里插入图片描述
ThreadLocal的set方法源码:

设置该线程局部变量的当前线程副本到指定的值。大多数子类都不需要这样做重写此方法,仅依赖于{@link #initialValue}方法设置线程局部变量的值。public void set(T value) {Thread t = Thread.currentThread();//ThreadLocalMap map = getMap(t);if (map != null) {map.set(this, value);} else {createMap(t, value);}}

在上面set方法里先获取到执行该方法的线程对象,再基于此对象去获取该线程的ThreadLocalMap 来实现线程间的隔离数据
getMap

获取与ThreadLocal相关联的映射
ThreadLocalMap getMap(Thread t) {return t.threadLocals;}

createMap:创建该线程的ThreadLocalMap

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

new ThreadLocalMap(this, firstValue)

ThreadLocalMap(ThreadLocal<?> firstKey, Object firstValue) {table = new Entry[INITIAL_CAPACITY];int i = firstKey.threadLocalHashCode & (INITIAL_CAPACITY - 1);table[i] = new Entry(firstKey, firstValue);size = 1;setThreshold(INITIAL_CAPACITY);}

**set方法执行结论:**先获取到执行该方法的线程对象,然后基于这个对象获取ThreadLocalMap,进行实例化,赋值。赋值的key就是ThreadLocal,虽然ThreadLocal在多线程中复用,但是实际存储数据的是ThreadLocalMap,这个ThreadLocalMap是和Thread绑定的

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();}

getMap

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

setInitialValue

private T setInitialValue() {T value = initialValue();Thread t = Thread.currentThread();ThreadLocalMap map = getMap(t);if (map != null) {map.set(this, value);} else {createMap(t, value);}if (this instanceof TerminatingThreadLocal) {TerminatingThreadLocal.register((TerminatingThreadLocal<?>) this);}return value;}

get方法执行结论: 先拿到执行方法的当前线程 根据当前线程去获取map,有就以ThreadLocal为key去获取value。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/pingmian/89197.shtml
繁体地址,请注明出处:http://hk.pswp.cn/pingmian/89197.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

CPO:对比偏好优化—突破大型语言模型在机器翻译中的性能边界

温馨提示&#xff1a; 本篇文章已同步至"AI专题精讲" CPO&#xff1a;对比偏好优化—突破大型语言模型在机器翻译中的性能边界 摘要 中等规模的大型语言模型&#xff08;LLMs&#xff09;&#xff0c;如参数量为 7B 或 13B 的模型&#xff0c;在机器翻译&#xff0…

执行shell 脚本 如何将日志全部输出到文件

在执行 Shell 脚本时&#xff0c;如果需要将 所有输出&#xff08;包括标准输出 stdout 和错误输出 stderr&#xff09; 重定向到日志文件&#xff0c;可以使用以下方法&#xff1a;方法 1&#xff1a;直接重定向&#xff08;推荐&#xff09; /appdata/mysql_backup_dump.sh &…

Postman接口测试实现UI自动化测试

Selenium底层原理 3天精通Postman接口测试&#xff0c;全套项目实战教程&#xff01;&#xff01;运行代码&#xff0c;启动浏览器后&#xff0c;webdriver会将浏览器绑定到特定的端口&#xff0c;作为webdriver的remote server&#xff08;远程服务端&#xff09;&#xff0c;…

CSS动画与变换全解析:从原理到性能优化的深度指南

引言&#xff1a;现代Web动画的技术革命 在当今的Web体验中&#xff0c;流畅的动画效果已成为用户交互的核心要素。根据Google的研究&#xff0c;60fps的动画可以使用户参与度提升53%&#xff0c;而卡顿的界面会导致跳出率增加40%。本文将深入剖析CSS动画&#xff08;animation…

NPM组件 @ivy-shared-components/iconslibrary 等窃取主机敏感信息

【高危】NPM组件 ivy-shared-components/iconslibrary 等窃取主机敏感信息 漏洞描述 当用户安装受影响版本的 ivy-shared-components/iconslibrary 等NPM组件包时会窃取用户的主机名、用户名、工作目录、IP地址等信息并发送到攻击者可控的服务器地址。 MPS编号MPS-zh19-e78w…

Fail2ban防止暴力破解工具使用教程

Fail2ban防止暴力破解工具使用教程场景Fail2ban安装和配置安装配置原理遇到的问题以及解决办法问题1&#xff1a;设置的策略是10分钟内ssh连接失败2次的ip进行封禁&#xff0c;日志中实际却出现4次连接。问题2&#xff1a;策略设置为1分钟内失败两次&#xff0c;封禁ip。但通过…

亚远景科技助力长城汽车,开启智能研发新征程

亚远景科技助力长城汽车&#xff0c;开启智能研发新征程在汽车智能化飞速发展的当下&#xff0c;软件研发管理成为车企决胜未来的关键。近日&#xff0c;亚远景科技胡浩老师应邀为长城汽车开展了一场主题深刻且极具实用价值的培训。本次培训聚焦软件研发管理导论 - 建立机器学习…

图算法在前端的复杂交互

引言 图算法是处理复杂关系和交互的强大工具&#xff0c;在前端开发中有着广泛应用。从社交网络的推荐系统到流程图编辑器的路径优化&#xff0c;再到权限依赖的拓扑排序&#xff0c;图算法能够高效解决数据之间的复杂关联问题。随着 Web 应用交互复杂度的增加&#xff0c;如实…

Prometheus Operator:Kubernetes 监控自动化实践

在云原生时代&#xff0c;Kubernetes 已成为容器编排的事实标准。然而&#xff0c;在高度动态的 Kubernetes 环境中&#xff0c;传统的监控工具往往难以跟上服务的快速变化。Prometheus Operator 应运而生&#xff0c;它将 Prometheus 及其生态系统与 Kubernetes 深度融合&…

一种融合人工智能与图像处理的发票OCR技术,将人力从繁琐的票据处理中解放

在数字化浪潮席卷全球的今天&#xff0c;发票OCR技术正悄然改变着企业财务流程的运作模式。这项融合了人工智能与图像处理的前沿技术&#xff0c;已成为财务自动化不可或缺的核心引擎。核心技术&#xff1a;OCR驱动的智能识别引擎发票OCR技术的核心在于光学字符识别&#xff08…

时空大数据:数字时代的“时空罗盘“

引言&#xff1a;为何需要“时空大数据”&#xff1f;“大数据”早已成为热词&#xff0c;但“时空大数据”的提出却暗含深刻逻辑。中国工程院王家耀院士指出&#xff0c;早期社会存在三大认知局限&#xff1a;过度关注商业大数据而忽视科学决策需求&#xff1b;忽视数据的时空…

PySide笔记之信号连接信号

PySide笔记之信号连接信号code review! 在 PySide6&#xff08;以及 Qt 的其他绑定&#xff0c;如 PyQt&#xff09;中&#xff0c;信号可以连接到信号。也就是说&#xff0c;可以把一个信号的发射&#xff0c;作为另一个信号的触发条件。这样做的效果是&#xff1a;当第一个信…

Linux操作系统之线程:线程概念

目录 前言&#xff1a; 一、进程与线程 二、线程初体验 三、分页式存储管理初谈 总结&#xff1a; 前言&#xff1a; 大家好啊&#xff0c;今天我们就要开始翻阅我们linux操作系统的另外一座大山&#xff1a;线程了。 对于线程&#xff0c;大体结构上我们是划分为两部分…

windows利用wsl安装qemu

首先需要安装wsl,然后在swl中启动一个子系统。这里我启动一个ubuntu22.04。 接下来的操作全部为在子系统中的操作。 检查虚拟化 在开始安装之前,让我们检查一下你的机器是否支持虚拟化。 要做到这一点,请使用以下命令: sean@DESKTOP-PPNPJJ3:~$ LC_ALL=C lscpu | grep …

如何使用 OpenCV 打开指定摄像头

在计算机视觉应用中&#xff0c;经常需要从特定的摄像头设备获取视频流。例如&#xff0c;在多摄像头环境中&#xff0c;当使用 OpenCV 的 cv::VideoCapture 类打开摄像头时&#xff0c;如果不指定摄像头的 ID&#xff0c;可能会随机打开系统中的某个摄像头&#xff0c;或者按照…

JAVA面试宝典 -《分布式ID生成器:Snowflake优化变种》

&#x1f680; 分布式ID生成器&#xff1a;Snowflake优化变种 一场订单高峰&#xff0c;一次链路追踪&#xff0c;一条消息投递…你是否想过&#xff0c;它们背后都依赖着一个“低调却关键”的存在——唯一ID。本文将带你深入理解分布式ID生成器的核心原理与工程实践&#xff0…

苹果的机器学习框架将支持英伟达的CUDA平台

苹果专为Apple Silicon设计的MLX机器学习框架即将迎来CUDA后端支持&#xff0c;这意义重大。原因如下。 这项开发工作由GitHub开发者zcbenz主导&#xff08;据AppleInsider报道&#xff09;&#xff0c;他于数月前开始构建CUDA支持的原型。此后他将项目拆分为多个模块&#xff…

golang语法-----变量、常量

变量1、声明与初始化&#xff08;1&#xff09;标准声明 (先声明&#xff0c;后赋值)var age int // 声明一个 int 类型的变量 age&#xff0c;此时 age 的值是 0 fmt.Println(age) // 输出: 0age 30 // 给 age 赋值 fmt.Println(age) // 输出: 30//int 的零…

Jenkins+Docker(docker-compose、Dockerfile)+Gitee实现自动化部署

项目目录结构 project-root/ ├── pom.xml ├── docker │ ├── copy.sh │ ├── file │ │ ├── jar │ │ │ └── 存放执行copy.sh以后jar包的位置 │ │ └── Dockerfile │ └── docker-compose.yml ├── docker-only-test │ ├─…

TASK01【datawhale组队学习】地瓜机器人具身智能概述

https://github.com/datawhalechina/ai-hardware-robotics 参考资料地址 具身智能&#xff08;Embodied AI&#xff09; 具身智能 智能的大脑 行动的身体。 比例&#xff08;Proportional&#xff09;、积分&#xff08;Integral&#xff09;、微分&#xff08;Derivative&a…