多线程(46)线程局部存储

线程局部存储(Thread Local Storage, TLS)是一种允许数据在多个线程中被独立地存储的编程范式。在Java中,这通过ThreadLocal类实现,它提供了一种线程封闭的机制,确保每个线程都有自己的变量副本,从而避免了变量共享所带来的线程安全问题。

工作原理

ThreadLocal创建的变量,对于使用它的每个线程,都会提供一个独立初始化的副本,各个线程可以修改自己的副本而不会影响其他线程的副本。这种机制特别适合于实现线程安全的数据格式,或者保存线程的上下文信息,避免了同步操作的性能损耗。

实现机制

ThreadLocal内部通过维护一个ThreadLocal.ThreadLocalMap来实现线程局部存储,这是一个定制化的哈希映射,用于存储每个线程的局部变量。每个Thread对象都有一个ThreadLocalMap的引用,但这个映射表只能通过ThreadLocal对象访问。

ThreadLocalget()set(T value)方法被调用时,会先找到当前线程的ThreadLocalMap,然后根据ThreadLocal对象作为键在这个映射表中查找或修改对应线程的局部变量。

核心源码

考虑到详细的源码分析可能非常复杂,这里提供一个简化的视角来理解ThreadLocal的核心逻辑:

public class ThreadLocal<T> {public T get() {Thread t = Thread.currentThread();ThreadLocalMap map = t.threadLocals;if (map != null) {ThreadLocalMap.Entry e = map.getEntry(this);if (e != null) {return (T)e.value;}}return setInitialValue();}public void set(T value) {Thread t = Thread.currentThread();ThreadLocalMap map = t.threadLocals;if (map != null) {map.set(this, value);} else {createMap(t, value);}}T initialValue() {return null;}private T setInitialValue() {T value = initialValue();Thread t = Thread.currentThread();ThreadLocalMap map = t.threadLocals;if (map != null) {map.set(this, value);} else {createMap(t, value);}return value;}// 注意:实际源码中ThreadLocalMap的实现更加复杂,这里仅为了提供一个直观的印象。static class ThreadLocalMap {static class Entry extends WeakReference<ThreadLocal<?>> {Object value;Entry(ThreadLocal<?> k, Object value) {super(k);this.value = value;}}private Entry[] table;void set(ThreadLocal<?> key, Object value) {// 实现省略:在表中找到或创建与key关联的条目,并更新其值}Entry getEntry(ThreadLocal<?> key) {// 实现省略:根据key找到对应的条目return null;}}void createMap(Thread t, T firstValue) {t.threadLocals = new ThreadLocalMap();t.threadLocals.set(this, firstValue);}
}

代码演示

下面是一个使用ThreadLocal的基本示例:

public class ThreadLocalExample {public static ThreadLocal<Integer> threadLocalValue = ThreadLocal.withInitial(() -> 1);public static void main(String[] args) throws InterruptedException {for (int i = 0; i < 3; i++) {final int threadId = i;new Thread(() -> {Integer value = threadLocalValue.get();System.out.println("Thread " + threadId + " initial value: " + value);// 修改本线程的threadLocalValuethreadLocalValue.set(value + threadId);// 再次读取,显示已修改的值System.out.println("Thread " + threadId + " new value: " + threadLocalValue.get());}).start();}}
}

在这个例子中,每个线程都能独立地获取和设置threadLocalValue的值,而不会影响到其他线程。

注意事项

虽然ThreadLocal可以方便地实现线程局部存储,但它也可能导致内存泄漏问题。因为ThreadLocal.ThreadLocalMap中的键(ThreadLocal对象)是通过弱引用实现的,而值(存储的对象)是通过强引用实现的。如果一个ThreadLocal对象没有任何强引用指向它,那么在下一次垃圾回收时,这个ThreadLocal对象将被回收,但是它在ThreadLocalMap中对应的值不会被回收,这就可能导致内存泄漏。为了避免这种情况,一般建议在不再需要使用ThreadLocal变量时调用其remove()方法。

总之,ThreadLocal提供了一种优雅的线程局部存储方案,但需要谨慎使用,以避免内存泄漏等问题。

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

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

相关文章

MongoDB聚合运算符:$objectToArray

文章目录 语法使用例1&#xff0c;无内嵌文档的情况例2&#xff0c;有内嵌文档的情况 举例objectToAarray 举例使用objectToArray对内嵌字段求和 o b j e c t T o A r r a y 与 objectToArray与 objectToArray与arrayToObject一起使用的例子 $objectToArray聚合运算符用于将文档…

绝地求生:PUBG×杜卡迪联名上线!参与投稿评论赢取精美好礼

PUBG杜卡迪联名活动游戏内现已正式上线&#xff01;我们诚邀与您一起在开拓未知战场和书写新历史的过程中&#xff0c;与杜卡迪一同实现您的极速梦想&#xff01; 在本次的杜卡迪工坊中&#xff0c;更是包含了具备标志性红色在内的6种颜色供您自由选择&#xff0c;一起自由驰骋…

Redis入门到通过之Redis安装

文章目录 Redis安装说明1.单机安装Redis1.1.安装Redis依赖1.2.上传安装包并解压1.3.启动1.3.1.默认启动1.3.2.指定配置启动1.3.3.开机自启 2.Redis客户端2.1.Redis命令行客户端2.2.图形化桌面客户端2.2.1.安装2.2.2.建立连接 Redis安装说明 大多数企业都是基于Linux服务器来部…

【Python 小学低段竞赛数学题】数字5在书本页码中出现16次,这本书最多有多少页

书的页码编号是按1 2 3 4 5一直这样下去的&#xff0c;数字5恰好出现了16次&#xff0c;请问这本书最多可以有多少页&#xff1f; 包含5的页码&#xff1a; 5 15 25 35 45 50 51 52 53 54 55 56 57 58 59 65 需要注意页码55&#xff0c;出现了两次5。因此到59页就满足数字5出现…

力扣爆刷第117天之CodeTop100五连刷71-75

力扣爆刷第117天之CodeTop100五连刷71-75 文章目录 力扣爆刷第117天之CodeTop100五连刷71-75一、48. 旋转图像二、39. 组合总和三、113. 路径总和 II四、34. 在排序数组中查找元素的第一个和最后一个位置五、394. 字符串解码 一、48. 旋转图像 题目链接&#xff1a;https://le…

Swift中的布尔型

在Swift中&#xff0c;布尔型数据用Bool类型表示。布尔型数据只有两个可能的值&#xff1a;true和false。布尔型数据通常用于条件判断和逻辑运算。例如&#xff1a; let isTrue true let isFalse falseif isTrue {print("这是真的") } else {print("这是假的…

GPT中的Transformer架构以及Transformer 中的注意力机制

目录 1 GPT中的Transformer架构 2 transformer中的注意力机制 参考文献&#xff1a; 看了两个比较好的视频&#xff0c;简单做了下笔记。 1 GPT中的Transformer架构 GPT是Generative Pre-trained Transformer单词的缩写&#xff0c;其中transformer是一种特定的神经网络&a…

如何排查k8s集群中Pod内mysqld进程占用内存消耗过高?

文章目录 1. **查看容器资源使用情况**&#xff1a;2. **进入容器内部**&#xff1a;3. **检查进程内存使用**&#xff1a;4. **MySQL服务器状态检查**&#xff1a;5. **MySQL日志分析**&#xff1a;6. **使用专门的MySQL监控工具**&#xff1a;7. **配置文件检查**&#xff1a…

Java基础07--多线程-网络编程-Java高级

一、多线程 1.认识多线程 ①线程 ②多线程 2.创建线程方式 ①方式一&#xff1a;继承Thread类 1.让子类继承Thread线程类 2.重写run方法&#xff0c;就是这个线程执行会执行的操作。 3.创建继承Thread的子类对象就代表一个线程 4.启动线程:.start()-自动执行run方法 注意&am…

全量知识系统 程序详细设计之 统一资产模型(QA-SmartChat)

Q1. 下面我们聊聊整个全知系统的设计 的矩阵和函数&#xff0c;矩阵表示的是“活物”&#xff0c;分别 类似 一个基因的活性、一个实体的辨识度和某种特征的可区分度。 函数的可微、可积和可导性 则表示 运动的控制方式 在全知系统设计中&#xff0c;矩阵和函数是两个核心的组…

Java中的Set、List、Map的区别及主要实现类方法

Java中的Set、List、Map的区别 数组是大小固定的&#xff0c;并且同一个数组只能存放类型一样的数据&#xff08;基本类型/引用类型&#xff09;&#xff0c;JAVA集合可以存储和操作数目不固定的一组数据。 所有的JAVA集合都位于 java.util包中&#xff01; JAVA集合只能存放引…

Linux netstat命令教程:网络统计工具(附实例详解和注意事项)

Linux netstat命令介绍 netstat&#xff08;网络统计&#xff09;是一个在Linux中用于显示网络相关信息并诊断各种网络问题的命令。它可以显示你的计算机正在进行的连接、发送信息的路径&#xff0c;甚至一些技术细节&#xff0c;如正在发送或接收的数据包的数量。 Linux net…

速盾:CDN是怎么防止ddos攻击的?

CDN&#xff08;内容分发网络&#xff09;是一种用于提高网站性能和安全性的重要技术。它通过在全球多个位置分布节点来存储和分发网站内容&#xff0c;以减少用户访问网站时的延迟并提高网站的可用性。除了提供高质量的内容分发&#xff0c;CDN还能有效地防止DDoS攻击。 DDoS…

Frida 远程RPC 调用进阶

引言: 今天讲下Frida 远程RPC 调用,为什么要用它,方便快捷。 安卓IOS 简单适用,代码量少很多,比Xposed。 1,安卓调用: 安卓调用,一般我们使用usb,模拟器的话直接用adb 命令去直连都可以,但是我这里叫一个稳定的方法,使用wifi adb ,比较靠谱,设置模拟器的ip,然…

点亮一颗 LED: 单片机 ch32v003 (RISC-V) 使用 rust 编写固件

首发日期 2024-04-09, 以下为原文内容: 使用 rust 编写单片机的程序 ? 很新, 但没问题. 使用 RISC-V CPU 的单片机 (比如 ch32v003) ? 也没问题. 同时使用 ? 哦嚯, 问题出现了 !! ch32v003 是一款使用 rv32ec 指令集的国产单片机, 很便宜 (某宝零卖只要 0.4 元一个, 在同档…

简单了解JVM

一.JVM简介 jvm及Java virtual machineJava虚拟机&#xff0c;它是一个虚构出来的计算机&#xff0c;一种规范。其实抛开这么专业的句子不说&#xff0c;就知道 JVM 其实就类似于一台小电脑运行在 windows 或者 linux 这些操作系统环境下即可。它直接和操作系统进行交互&#…

Handler——小白能懂的原理,老鸟需要的面经

1.机制学习 1.1Handler定义 发送并处理 与线程的消息队列关联的Message和Runnable 1.2基本用法 1、Message.obtain() 从消息池取得Message 2、Handler().sendMessage(msg) 发送消息 3、Handler().post 将Runnable包装成Message发送 以下提供一个结构代码 import android.os.H…

VMware导出虚拟机vmkd格式转换qcow2

VMware虚拟机导出qcow2格式可以上传至云服务 1、需要导出的虚拟机 2、克隆虚拟机 3、选择克隆源 4、创建完整克隆 5、完成 6、找到VMware安装路径 7、找到vmware-vdiskmanager所在路径使用cmd或Windows PowerShell进入目录 进入vmware-vdiskmanager目录 cd F:\软件\VMware Wo…

VRTK/SteamVR手柄震动功能

VRTK/SteamVR手柄震动功能 前言代码块 前言 手柄震动功能配合虚拟仿真模块的模拟电击等功能非常方便 代码块 SteamVR_Controller.DeviceRelation.Rightmost是右侧手柄 SteamVR_Controller.DeviceRelation.Leftmost是左侧手柄 var deviceIndex2 SteamVR_Controller.GetDevic…

debian安装和基本使用

debian安装和基本使用 文章目录 debian安装和基本使用1. 为什么选择debian2. 如何下载Debian2.1 小型安装镜像2.2 完整安装镜像 3. Debian操作系统安装3.1 创建Debian虚拟机3.2 安装操作系统 4. Debian系统的初始设置4.1 桌面环境的配置4.2 配置网络4.3 生效网络配置4.4 配置de…