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

}

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

PS从入门到精通视频各类教程整理全集,包含素材、作业等(4)

PS从入门到精通视频各类教程整理全集&#xff0c;包含素材、作业等 最新PS以及插件合集&#xff0c;可在我以往文章中找到 由于阿里云盘有分享次受限制和文件大小限制&#xff0c;今天先分享到这里&#xff0c;后续持续更新 PS人物数码照片处理技法视频教程 https://www.…

武汉星起航:一站式跨境电商服务引领者,专业高效助力客户出海

武汉星起航电子商务有限公司&#xff0c;坐落于华中地区的商业核心地带——湖北武汉&#xff0c;自公司成立以来&#xff0c;便以提供一站式跨境电商服务为核心发展&#xff0c;致力于为广大客户提供专业、高效、全面的出海解决方案。凭借5对1服务体系、ERP软件授权、中转仓服务…

二、分布式事务

目录 二、分布式事务2.1 什么是分布式事务2.2 分布式事务产生的背景2.3 分布式事务产生的场景2.4 分布式事务理论4.1 CAP理论4.2 Base理论 5、分布式事务的解决方案 二、分布式事务 2.1 什么是分布式事务 一组操作会产⽣多个数据库session会话 此时就会出现分布式事务 2.2 分…

游戏软件出现d3dcompiler_47.dll缺失怎么修复,亲测的六种有效方法推荐

D3DCompiler47.dll是DirectX SDK中的一个重要组件&#xff0c;它提供了将HLSL&#xff08;High-Level Shading Language&#xff09;着色器编译为可执行代码的功能。通过使用D3DCompiler47.dll&#xff0c;开发人员可以将复杂的着色器代码转换为可以在GPU上高效运行的机器代码&…

黑马点评项目笔记 II

基于Stream的消息队列 stream是一种数据类型&#xff0c;可以实现一个功能非常完善的消息队列 key&#xff1a;队列名称 nomkstream&#xff1a;如果队列不存在是否自动创建&#xff0c;默认创建 maxlen/minid&#xff1a;设置消息队列的最大消息数量 *|ID 唯一id&#xff1a;…

Vue系列-el挂载

<!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>el:挂载点</title> </head> <body&g…

作业 二维数组-定位问题

图形相似度 描述 给出两幅相同大小的黑白图像&#xff08;用0-1矩阵&#xff09;表示&#xff0c;求它们的相似度。 说明&#xff1a;若两幅图像在相同位置上的像素点颜色相同&#xff0c;则称它们在该位置具有相同的像素点。 两幅图像的相似度定义为相同像素点数占总像素点数…

P87 4.1 C++ FOR 与Delphi FOR 的区别

输出x, sin(x), cos(x), tan(x)的值。已知X0&#xff0c;10&#xff0c; 20&#xff0c;180。 我用Delphi编写了程序&#xff1a; 第10行出现 给FOR 循环变量赋值i错误。 C中是可以的&#xff0c; 详见&#xff1a;delphi循环的一个小知识_assignment to for-loop variable…

安装JupyterLab的集成环境

Python集成环境安装 不要半途而废&#xff0c;不要作业太多就抛下你手中的笔&#xff0c;拿起你旁边的手机&#xff0c;你觉得这样很有意义吗&#xff1f;一个小时一道题都没做&#xff0c;盯着手机屏幕它能给你一个未来吗&#xff1f;少分心就能多做一道题&#xff0c;多学样本…

Java多线程:定位死锁

检测死锁可以使用jconsole工具&#xff0c;或使用jps定位进程id&#xff0c;再用jstack定位死锁 方案1&#xff1a; 1. 先用jps查看所有的java进程id 2. jstack 进程id定位死锁 3. 查看死锁结果 方案2:从jdk的安装路径中找到bin目录, 点击jconsole

Kafka入门到实战-第五弹

Kafka入门到实战 Kafka常见操作官网地址Kafka概述Kafka的基础操作更新计划 Kafka常见操作 官网地址 声明: 由于操作系统, 版本更新等原因, 文章所列内容不一定100%复现, 还要以官方信息为准 https://kafka.apache.org/Kafka概述 Apache Kafka 是一个开源的分布式事件流平台&…

1.5编写一个程序,输入梯形的上底,下底和高,输出梯形的面积。

1、编写一个程序,输入梯形的上底,下底和高,输出梯形的面积。 package com.kangning.web.controller.system;import java.util.Scanner;/*** 编写一个程序,输入梯形的上底,下底和高,输出梯形的面积。*/ public class CountArea {public static void main(String[] args) …

知乎:多云架构下大模型训练,如何保障存储稳定性?

知乎&#xff0c;中文互联网领域领先的问答社区和原创内容平台&#xff0c;2011 年 1 月正式上线&#xff0c;月活跃用户超过 1 亿。平台的搜索和推荐服务得益于先进的 AI 算法&#xff0c;数百名算法工程师基于数据平台和机器学习平台进行海量数据处理和算法训练任务。 为了提…

java入门学习Day01

本篇文章主要是学会如何使用IDEA&#xff0c;和运行第一个java文件。 java环境安装&#xff1a;Windows下Java环境配置教程_windows java环境配置-CSDN博客 IDEA安装&#xff1a;IDEA 2023.2.5 最新激活码,注册码&#xff08;亲测好用&#xff09; - 异常教程 以上两个链接…

函数栈帧的创建与销毁(最详细的一集)上

前言 1.我们在进行c语言代码编程的时候&#xff0c;常常会把独立的一个功能抽象为函数&#xff0c;利用函数去实现各种的功能&#xff0c;那么&#xff0c;函数是如何调用的&#xff1f;函数的返回值是怎么返回的&#xff1f;参数又是如何传参的&#xff1f;所有这些问题都会跟…

【NLP练习】Pytorch文本分类入门

Pytorch文本分类入门 &#x1f368; 本文为&#x1f517;365天深度学习训练营 中的学习记录博客 &#x1f356; 原作者&#xff1a;K同学啊 | 接辅导、项目定制一、前期准备 1. 环境安装 确保已经安装torchtext与portalocker库 2. 加载数据 #加载数据 import torch import t…

【滑动窗口】Leetcode 找到字符串中所有字母异位词

题目解析 438. 找到字符串中所有字母异位词 算法讲解 寻找目标串的异位词&#xff0c;我们使用固定长度的滑动窗口&#xff0c;首先我们判断窗口左右的字符是否存在于目标串中&#xff0c;如果不存在就让窗口滑动&#xff1b;存在的话&#xff0c;我们就把字符丢进Hash中&a…

【JavaSE】类和对象详解(上)

欢迎关注个人主页&#xff1a;逸狼 创造不易&#xff0c;可以点点赞吗~ 如有错误&#xff0c;欢迎指出~ 目录 类和对象 类的组成 对类的理解 成员变量的访问和类方法的调用 this 抛出一个问题 this的作用 初始化成员变量 未初始化的成员变量 代码举例 就地初始化 构…

每日一题(leetcode331):验证二叉树的前序序列化——栈

类似消消乐&#xff0c;数字&#xff0b;“#”“#”就可以消成一个“#”&#xff0c;到最后如果栈中只剩一个“#”便说明序列正确。 当然也可以用槽位理解&#xff0c;一个数字出现会消耗一个槽位产生两个槽位&#xff08;即产生一个槽位&#xff09;&#xff0c;一个“#”出现…

题目:安全序列(蓝桥OJ 3432)

问题描述&#xff1a; 题解&#xff1a; #include <bits/stdc.h> using namespace std; using ll long long; const int N 1e6 9, p 1e9 7;int prefix[N],dp[N];int main() {int n, k;cin >> n >> k;dp[0] prefix[0] 1;for(int i 1; i < n; i){i…