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…

navicat12版本重置试用期时间

一、步骤 关闭NavicatWin R&#xff0c;输入regedit回车删除HKEY_CURRENT_USER\Software\PremiumSoft\Data删除HKEY_CURRENT_USER\Software\Classes\CLSID 二、这注册表直接删了&#xff0c;没什么安全问题 删了方便&#xff0c;只能这么说HKEY_CURRENT_USER\Software\Prem…

正弦实时数据库的应用(1)-数字孪生

前面说过实时数据库比较合适应用在数字孪生系统上&#xff0c;本篇就来对比一下数字孪生系统各种数据存储的应用方案的优缺点&#xff0c;几种典型的数据存储方式如下所示&#xff1a; 方案一&#xff1a;RedisMySQL/MongoDB方案二&#xff1a;MongoDB方案三&#xff1a;时序数…

算法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…

JQuery入门基础

文章目录 概述DOM对象与jQuery对象的转换如何解决jQuery与其他JS库共存问题($符号)常用API选择器(9大类)1、基本选择器2、层级选择器3、属性选择器4、基本过滤选择器5、内容过滤选择器6、可见性过滤选择器7、子元素过滤选择器8、表单过滤选择器(用处不大)9、表单对象滤选择…

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

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

故障诊断模型 | 基于交叉注意力融合时频特征的轴承故障诊断模型

基于交叉注意力融合时频特征的轴承故障诊断模型是一种先进的诊断方法,结合了信号处理、深度学习和注意力机制等多种技术,以提高轴承故障识别的准确性和效率。 一、模型概述 该模型主要利用交叉注意力机制融合时域和频域的特征,通过深度学习算法对轴承的振动信号进行处理和…

js中的while循环和do while循环的区别

在 JavaScript 中&#xff0c;while 循环和 do...while 循环都是用来重复执行一段代码块&#xff0c;直到满足某个条件为止。它们的主要区别在于条件检查的位置。 while 循环&#xff1a;在 while 循环中&#xff0c;条件检查在循环体的开始之前进行。如果条件为真&#xff0c…

后管配置js

const chalk require(chalk) const pathrequire(path) function resolve(dir){ return path.join(__dirname,dir) } module.exports{ //没有书写outputDir属性 默认dist 对应dev.assetSubDirectory outputDir:navigation_WEB_new //http://vue.org/v2/guide/installti…

服了,一线城市的后端都卷成这样了吗!?

声明&#xff1a;本文首发在同名公众号&#xff1a;王中阳Go&#xff0c;未经授权禁止转载。 先听TA的故事 投稿的主人公是一名工作5年的后端开发工程师&#xff0c;最近2年用Golang&#xff0c;之前其他语言。去年春节前被裁员了&#xff0c;各种心酸史&#xff0c;好愁人啊。…

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;也不会保证能够取出多少或者能够把数据都取完。比如斐波那契额数列&#…

【IntermLM2】学习笔记

微调方式 在大模型的下游应用中&#xff0c;可以有两种微调方式 增量续训 即无监督的方式&#xff0c;让模型学习一些新知识&#xff0c;比如某些垂直领域的新知识 使用的数据有&#xff1a;书籍&#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;分享嵌入式开发领域的相关知识、经验、思考和感悟,欢迎关注。提供嵌入式方向的学习指导…

在Compose中方便的使用MVI思想?试试useReducer!

写在前面 本文中提及的use开头的函数&#xff0c;都出自与我的 ComposeHooks 项目&#xff0c;它提供了一系列 React Hooks 风格的状态封装函数&#xff0c;可以帮你更好的使用 Compose&#xff0c;无需关系复杂的状态管理&#xff0c;专心于业务与UI组件。 这是系列文章的第…

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()…