ThreadLocal的基本使用

一、ThreadLocal的介绍

  1. ThreadLocal 是 Java 中的一个类,它提供了线程局部变量的功能。线程局部变量是指每个线程拥有自己独立的变量副本,这些变量在不同的线程中互不影响。ThreadLocal 提供了一种在多线程环境下,每个线程都可以独立访问自己的变量副本的机制。
    • ThreadLocal并不是一个线程,而是一个线程的局部变量
    • ThreadLocal为每个线程提供单独的一份存储空间,具有线程隔离的效果,只有在线程内才能获取到对应的值,线程外则不能访问。
      在这里插入图片描述

二、ThreadLocal的使用场景

在springboot中,客户端每一次发送请求,tomcat服务器都会分配一个单独的线程,然后在这个线程上可能要执行不同的代码,比如controller、拦截器的代码啊,service的代码等等,它们都属于同一个线程。满足这个要求就可以使用ThreadLocal进行存储数据。(每一个线程都有一个单独的存储空间,那么在线程的生命周期内,我们可以共享这份存储空间)

  • ThreadLocal 在 Java 中有许多使用场景,主要用于解决以下两类问题:

      1. 线程封闭性(Thread Confinement):有些对象是非线程安全的,但是它们只在特定的线程中使用,并不被多个线程共享。使用 ThreadLocal 可以确保每个线程都有自己的对象副本,从而避免了线程安全问题。
      1. 避免传递参数(Avoid Passing Parameters):有些情况下,某个对象需要在一个方法调用链中的多个方法之间传递,但这个对象对于整个调用链来说是相同的,不应该在方法之间显式地传递。使用 ThreadLocal 可以将这个对象存储在线程局部变量中,而不必在方法参数中传递。
  • 一些典型的 ThreadLocal 使用场景包括:

    • 数据库连接管理:在多线程环境下,每个线程需要独立的数据库连接,可以使用 ThreadLocal 来管理每个线程的数据库连接,避免多个线程共享连接导致的线程安全问题。

    • 会话管理:在 Web 应用中,每个用户会话通常都需要存储用户的身份信息或者其他会话相关的数据,可以使用 ThreadLocal 来存储每个用户会话的信息,确保线程安全性。

    • 日志跟踪:在分布式系统中,通常需要在不同的服务之间传递某个请求的唯一标识符(例如请求ID),可以使用 ThreadLocal 来存储这个标识符,从而在整个请求处理过程中都可以方便地访问到它。

    • 线程池任务参数传递:在使用线程池执行任务时,有时候需要将一些任务相关的参数传递给执行任务的线程,可以使用 ThreadLocal 来存储这些参数,而不必在任务执行时显式地传递参数。

  • 总的来说,ThreadLocal 适合于需要在多个方法调用之间共享数据但又不希望使用方法参数显式传递数据的情况。但需要注意,滥用 ThreadLocal 会导致内存泄漏或者其他问题,因此在使用时需要谨慎

三、ThreadLocal的实现原理

  • ThreadLocal 的实现原理主要涉及 ThreadLocal 类本身以及 Thread 类的实现机制。
  1. ThreadLocal 类

    • ThreadLocal 内部维护了一个以当前线程为 key、存储的对象为 value 的 map,这个 map 是 ThreadLocal 的一个静态成员变量,被所有的 ThreadLocal 实例共享。
    • 每个 ThreadLocal 实例通过 get()、set() 方法与当前线程关联,通过当前线程获取或设置对应的值。
    • ThreadLocal 的 get() 方法首先获取当前线程,然后通过当前线程作为 key 在 map 中查找对应的值。
    • ThreadLocal 的 set() 方法首先获取当前线程,然后将当前线程与设置的值关联起来,存储在 map 中。
  2. Thread 类

    • Java 中的线程是由 Thread 类实现的。每个线程对象都会包含一个 ThreadLocalMap 类型的成员变量,用于存储线程局部变量。
    • 当调用 ThreadLocal 的 set() 方法时,实际上是将值存储到当前线程的 ThreadLocalMap 中,以 ThreadLocal 对象为键,值为值。
    • 当调用 ThreadLocal 的 get() 方法时,实际上是从当前线程的 ThreadLocalMap 中获取对应的值。

总的来说,ThreadLocal 的实现原理是通过在每个线程中维护一个 ThreadLocalMap 对象,这个对象中存储了当前线程所有 ThreadLocal 对象的键值对。通过这种方式,每个线程可以独立地访问自己的线程局部变量,而不会与其他线程产生冲突。需要注意的是,由于 ThreadLocalMap 是存储在每个线程中的,因此需要注意内存泄漏的问题,及时清理不再需要的线程局部变量是很重要的

四、ThreadLocal的基本使用

public class ThreadLocalExample {// 定义一个 ThreadLocal 变量private static ThreadLocal<String> threadLocal = new ThreadLocal<>();public static void main(String[] args) {// 在主线程中设置 ThreadLocal 的值threadLocal.set("Main Thread Value");// 创建一个新的线程,并在其中设置 ThreadLocal 的值Thread thread = new Thread(() -> {// 在子线程中获取并输出 ThreadLocal 的值System.out.println("ThreadLocal value in new thread: " + threadLocal.get());// 设置新的 ThreadLocal 值threadLocal.set("New Thread Value");// 再次输出新的 ThreadLocal 值System.out.println("Updated ThreadLocal value in new thread: " + threadLocal.get());});thread.start();// 主线程中获取并输出 ThreadLocal 的值System.out.println("ThreadLocal value in main thread: " + threadLocal.get());// 清除 ThreadLocal 的值threadLocal.remove();}
}

输出:

ThreadLocal value in main thread: Main Thread Value
ThreadLocal value in new thread: null
Updated ThreadLocal value in new thread: New Thread Value
  • 在这个示例中,主线程通过 threadLocal.set(“Main Thread Value”) 设置了一个 ThreadLocal 的值。
  • 然后创建了一个新的线程,在新线程中通过 threadLocal.get() 获取了这个值(发现获取不到,为null,因为不在同一个线程),并设置了新的值。
  • 在主线程和新线程中分别输出了 ThreadLocal 的值。主线程获取到的是一开始设置的值,而新线程获取到的是新线程里面设置的值。
  • 值得注意的是,在主线程和新线程中的 ThreadLocal 值是相互独立的,互不影响。最后通过threadLocal.remove() 清除了 ThreadLocal 的值。

这样就展示了 ThreadLocal 在不同线程中可以存储不同的数据副本,每个线程都可以独立地访问和修改自己的副本,而不会影响其他线程。

五、使用ThreadLocal的时候一般都会封装成一个工具类

public class BaseContext {public static ThreadLocal<Long> threadLocal = new ThreadLocal<>();public static void setCurrentId(Long id) {threadLocal.set(id);}public static Long getCurrentId() {return threadLocal.get();}public static void removeCurrentId() {threadLocal.remove();}}

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

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

相关文章

多叉树题目:N 叉树的最大深度

文章目录 题目标题和出处难度题目描述要求示例数据范围 解法一思路和算法代码复杂度分析 解法二思路和算法代码复杂度分析 题目 标题和出处 标题&#xff1a;N 叉树的最大深度 出处&#xff1a;559. N 叉树的最大深度 难度 3 级 题目描述 要求 给定一个 N 叉树&#xf…

算法6.4-6.6DFS

一个不知名大学生&#xff0c;江湖人称菜狗 original author: Jacky Li Email : 3435673055qq.com Time of completion&#xff1a;2024.03.27 Last edited: 2024.03.27 目录 算法6.4-6.6DFS 第1关&#xff1a;算法6.5采用邻接矩阵表示图的深搜 任务描述 相关知识 编程要求…

2024银行业最新数字化转型的方法与路径

银行业数字化转型是一场由思想到行动、由顶层到基层、由内部到外部的深刻变革&#xff0c; 需要科学方法论的指导。在推动体系性重塑、开放生态建设、业务科技融合、基础设施升 级以及体制机制变革等探索和实践中&#xff0c;银行业逐步形成从顶层设计到数字化能力建设&#xf…

【数据结构】非线性结构---二叉树

1、树 1.1 树的相关概念 节点的度&#xff1a;一个节点含有的子树的个数称为该节点的度&#xff1b; 如上图&#xff1a;A的为6 叶节点或终端节点&#xff1a;度为0的节点称为叶节点&#xff1b; 如上图&#xff1a;B、C、H、I...等节点为叶节点 非终端节点或分支节点&#…

docker部署实用的运维开发手册

下载镜像 docker pull registry.cn-beijing.aliyuncs.com/wuxingge123/reference:latestdocker-compose部署 vim docker-compose.yml version: 3 services:reference:container_name: referenceimage: registry.cn-beijing.aliyuncs.com/wuxingge123/reference:latestports:…

ES6学习之路:迭代器Iterator和生成器Generator

迭代器 一、知识背景 什么是迭代器 迭代器就是在一个数据集合中不断取出数据的过程迭代和遍历的区别 遍历是把所有数据都取出迭代器注重的是依次取出数据&#xff0c;它不会在意有多少数据&#xff0c;也不会保证能够取出多少或者能够把数据都取完。比如斐波那契额数列&#…

openstack云计算(二)——使用Packstack安装器安装一体化OpenStack云平台

初步掌握OpenStack快捷安装的方法。掌握OpenStack图形界面的基本操作。 一【准备阶段】 &#xff08;1&#xff09;准备一台能够安装OpenStack的实验用计算机&#xff0c;建议使用VMware虚拟机。 &#xff08;2&#xff09;该计算机应安装CentOS 7&#xff0c;建议采用CentO…

基于Sermant的全链路灰度发布在汽车行业DMS系统的应用

作者&#xff1a;聂子雄 华为云高级软件工程师 摘要 随着汽车产业的智能升级&#xff0c;DMS系统作为汽车行业的经销管理系统也面临着更加多种多样的业务场景的挑战。借助Sermant&#xff0c;华为云能够为DMS系统提供一整套端到端全链路灰度发布方案&#xff0c;这套方案可以…

深度学习训练中常用的三个基础库tqdmargparseyaml

文章目录 训练常用工具[tqdm][argparse][yaml]tqdm1. 导入tqdm2. 传入可迭代对象快速使用进阶1&#xff1a;通过update()自定义进度条每次更新的步长进阶2&#xff1a;通过set_description和set_postfix自定义进度条内容 Argparse第一步&#xff1a;创建ArgumentParser对象第二…

机器学习在智能音箱中的应用探索与实践:让声音更懂你

&#x1f9d1; 作者简介&#xff1a;阿里巴巴嵌入式技术专家&#xff0c;深耕嵌入式人工智能领域&#xff0c;具备多年的嵌入式硬件产品研发管理经验。 &#x1f4d2; 博客介绍&#xff1a;分享嵌入式开发领域的相关知识、经验、思考和感悟,欢迎关注。提供嵌入式方向的学习指导…

2024 ccfcsp认证打卡 2023 03 01 田地丈量

import java.util.Scanner;public class Main {public static void main(String[] args) {Scanner in new Scanner(System.in);int n in.nextInt(); // 输入 n&#xff0c;表示矩形的数量int a in.nextInt(); // 输入 a&#xff0c;表示整个区域的长度int b in.nextInt()…

Hive详解(5)

Hive 窗口函数 案例 需求&#xff1a;连续三天登陆的用户数据 步骤&#xff1a; -- 建表 create table logins (username string,log_date string ) row format delimited fields terminated by ; -- 加载数据 load data local inpath /opt/hive_data/login into table log…

如何在Portainer中创建Nginx服务并搭建静态站点实现公网访问本地网站

文章目录 前言1. 安装Portainer1.1 访问Portainer Web界面 2. 使用Portainer创建Nginx容器3. 将Web静态站点实现公网访问4. 配置Web站点公网访问地址4.1公网访问Web站点 5. 固定Web静态站点公网地址6. 固定公网地址访问Web静态站点 前言 Portainer是一个开源的Docker轻量级可视…

报错:TypeError: Cannot handle this data type: (1, 1, 3), <f8

报错内容&#xff1a; 解决方法&#xff1a; 这个错误是由于 PIL 库无法处理特定的数据类型引起的。为了解决这个问题&#xff0c;你可以尝试将数据类型转换为 PIL 可以处理的类型&#xff0c;比如转换为 uint8 类型。你可以在调用 Image.fromarray() 方法之前&#xff0c;将…

SQL,group by分组后分别计算组内不同值的数量

SQL&#xff0c;group by分组后分别计算组内不同值的数量 如现有一张购物表shopping 先要求小明和小红分别买了多少笔和多少橡皮&#xff0c;形成以下格式 SELECT name,COUNT(*) FROM shopping GROUP BY name;SELECT name AS 姓名,SUM( CASE WHEN cargo 笔 THEN 1 ELSE 0 END)…

使用CRXjs、Vite、Vue 开发 Chrome 多页面插件,手动配置 vite.config.ts 和 manifest.json 文件

一、使用CRXjs、Vite、Vue 开发 Chrome 多页面插件&#xff0c;手动配置 vite.config.ts 和 manifest.json 文件 一、创建 Vue 项目 1. 使用 Vite 创建 Vue 项目 npm create vitelatest # npm yarn create vite # yarn pnpm create vite # pnpm选择 Vue 和 TS 进入项目…

在Windows中使用NVM安装node.js

NVM介绍 Node.js版本管理器&#xff08;Node Version Manager&#xff09;&#xff0c;简称NVM&#xff0c;是一款用于在单个系统上轻松安装和管理多个Node.js版本的命令行工具。它允许用户根据项目需求在不同版本之间自由切换&#xff0c;解决了因为不同项目依赖于不同Node.j…

Python快速入门系列-6(Python高级特性)

第六章: Python高级特性 6.1 列表推导式与生成器6.1.1 列表推导式6.1.2 生成器6.1.2.1 生成器表达式6.1.2.2 生成器函数6.2 装饰器与迭代器6.2.1 装饰器6.2.2 迭代器6.3 异常处理与错误调试6.3.1 异常处理6.3.1.1 try-except语句6.3.1.2 try-except-else语句6.3.2 错误调试6.3…

【缺陷】硅光电二极管中的DT侧壁陷阱态的DLTS表征

【A DLTS study on Deep Trench Processing induced Trap States in Silicon Photodiodes】 概括 本研究通过深能级瞬态光谱&#xff08;DLTS&#xff09;技术对硅光电二极管中的深沟槽&#xff08;DT&#xff09;侧壁诱导的陷阱态进行了详细分析。研究发现&#xff0c;这些陷…

golang语言系列:通用技能之 Scrum、Kanban等敏捷管理策略

云原生学习路线导航页&#xff08;持续更新中&#xff09; 本文是 golang语言系列 文章&#xff0c;主要对编程通用技能 Scrum、Kanban等敏捷管理策略 进行学习 1.什么是敏捷开发 敏捷是一个描述软件开发方法的术语&#xff0c;它强调增量交付、团队协作、持续规划和持续学习。…