首页 > 其他分享 >ThreadLocal工作原理

ThreadLocal工作原理

时间:2023-03-22 16:02:14浏览次数:41  
标签:同步 变量 Object 工作 ThreadLocal 线程 原理 多线程


 1.概述

    ThreadLocal为我们解决多线程程序的并发问题提供了一种新的思路。使用这个工具类可以很简洁地编写出优美的多线程程序。
    ThreadLocal很容易让人望文生义,想当然地认为是一个“本地线程”。其实,ThreadLocal并不是一个Thread,而是Thread的局部变量,也许把它命名ThreadLocalVariable更容易让人理解一些。
    当使用ThreadLocal维护变量时,ThreadLocal为每个使用该变量的线程提供独立的变量副本,所以每一个线程都可以独立地改变自己的副本,而不会影响其它线程所对应的副本。

2.ThreadLocal的接口方法

    ThreadLocal类接口很简单,只有4个方法,我们先来了解一下:
void set(Object value)
    设置当前线程的线程局部变量的值。
public Object get()
    该方法返回当前线程所对应的线程局部变量。
public void remove()
    将当前线程局部变量的值删除,目的是为了减少内存的占用,该方法是JDK 5.0新增的方法。需要指出的是,当线程结束后,对应该线程的局部变量将自动被垃圾回收,所以显式调用该方法清除线程的局部变量并不是必须的操作,但它可以加快内存回收的速度。
protected Object initialValue()
    返回该线程局部变量的初始值,该方法是一个protected的方法,显然是为了让子类覆盖而设计的。这个方法是一个延迟调用方法,在线程第1次调用get()或set(Object)时才执行,并且仅执行1次。ThreadLocal中的缺省实现直接返回一个null。
    值得一提的是,在JDK5.0中,ThreadLocal已经支持泛型,该类的类名已经变为ThreadLocal<T>。API方法也相应进行了调整,新版本的API方法分别是void set(T value)、T get()以及T initialValue()。

3.实现原理

   ThreadLocal是如何做到为每一个线程维护变量的副本的呢?其实实现的思路很简单:在ThreadLocal类中有一个Map,用于存储每一个线程的变量副本,Map中元素的键为线程对象,而值对应线程的变量副本。我们自己就可以提供一个简单的实现版本:

1. public class ThreadLocal{   
2. private Map values =Collections.synchronizedMap(newHashMap()); 
3. public Object get(){
4.    Thread curThread = Thread.currentThread();
5.    Object o = values.get(curThread);
6. if (o == null||!values.containsKey(curThread)){
7.    o = initialValue();
8.     values.put(curThread, o);
9. }
10. return o; 
11. }
12.
13.
14. public void set(Object newValue){
15.    values.put(Thread.currentThread(), newValue);
16. }
17.
18.
19. public Object initialValue(){
20. return null;
21. }
22. }


4.Thread同步机制的比较

    ThreadLocal和线程同步机制相比有什么优势呢?ThreadLocal和线程同步机制都是为了解决多线程中相同变量的访问冲突问题。
    在同步机制中,通过对象的锁机制保证同一时间只有一个线程访问变量。这时该变量是多个线程共享的,使用同步机制要求程序慎密地分析什么时候对变量进行读写,什么时候需要锁定某个对象,什么时候释放对象锁等繁杂的问题,程序设计和编写难度相对较大。
    而ThreadLocal则从另一个角度来解决多线程的并发访问。ThreadLocal会为每一个线程提供一个独立的变量副本,从而隔离了多个线程对数据的访问冲突。因为每一个线程都拥有自己的变量副本,从而也就没有必要对该变量进行同步了。ThreadLocal提供了线程安全的共享对象,在编写多线程代码时,可以把不安全的变量封装进ThreadLocal。
由于ThreadLocal中可以持有任何类型的对象,低版本JDK所提供的get()返回的是Object对象,需要强制类型转换。但JDK 5.0通过泛型很好的解决了这个问题,在一定程度地简化ThreadLocal的使用,代码清单 9 2就使用了JDK 5.0新的ThreadLocal<T>版本。
    概括起来说,对于多线程资源共享的问题,同步机制采用了“以时间换空间”的方式,而ThreadLocal采用了“以空间换时间”的方式。前者仅提供一份变量,让不同的线程排队访问,而后者为每一个线程都提供了一份变量,因此可以同时访问而互不影响。
    我们知道在一般情况下,只有无状态的Bean才可以在多线程环境下共享,在Spring中,绝大部分Bean都可以声明为singleton作用域。就是因为Spring对一些Bean(如RequestContextHolder、TransactionSynchronizationManager、LocaleContextHolder等)中非线程安全状态采用ThreadLocal进行处理,让它们也成为线程安全的状态,因为有状态的Bean就可以在多线程中共享了。

5.小结

    当然ThreadLocal并不能替代同步机制,两者面向的问题领域不同。同步机制是为了同步多个线程对相同资源的并发访问,是为了多个线程之间进行通信的有效方式;而ThreadLocal是隔离多个线程的数据共享,从根本上就不在多个线程之间共享资源(变量),这样当然不需要对多个线程进行同步了。所以,如果你需要进行多个线程之间进行通信,则使用同步机制;如果需要隔离多个线程之间的共享冲突,可以使用ThreadLocal,这将极大地简化你的程序,使程序更加易读、简洁。  

标签:同步,变量,Object,工作,ThreadLocal,线程,原理,多线程
From: https://blog.51cto.com/u_2650279/6142857

相关文章

  • 云原生服务网格Istio:原理、实践、架构与源码解析
    华为云原生团队600多页的Istio实战精华总结,云原生服务网格Istio:原理、实践、架构与源码解析的电子书。图书介绍《云原生服务网格Istio:原理、实践、架构与源码解析》分......
  • Git 记录在非工作分支中 做了开发工作 - 挽回
    之前切到了测试分支“develop”,由于改bug比较急一时忘记了是在哪个分支的事。当要推送到自己的分支的时候突然发现目前所有的工作竟然在测试分支上,我的天啊一时懵了,想......
  • 解决Kafka总是丢消息的方法和原理
    注:本文转自:https://www.toutiao.com/article/7210953985497678347/?log_from=f0ecce317abb8_1679450040551引入MQ消息中间件最直接的目的:系统解耦以及流量控制(削峰填谷)......
  • 测试工程师的工作
    theme:channing-cyan测试这份工作确实入门相对来说是比较低的,看起来谁都能做,但是不一定谁都能做好,并且在一些小公司里面,测试往往还会被其他人认为是没有技术含量的工作,......
  • 为工作排好优先级
    工作,是干不完的,因此我们需要分清轻重缓急,为它们划分优先级,这样才不至于让自己手忙脚乱。给手头的事情排上正确的优先级,是一项很重要的工作能力。优先级有很多考量,并不是......
  • Android数据结构-SparseArray实现原理
    SparseArray家族SparseArray基于键值对存储数据,key为int,value为object,简单使用如下://声明SparseArray<String>sparseArray=newSparseArray<>();......
  • CVPR 2023 | GPT-4与文心一言同台竞技,居然是为了自动驾驶UniAD工作!
    以下文章来源于OpenDriveLab ,作者OpenDriveLa00  前言都说ChatGPT是自然语言处理中技术大魔王,国内百度的文心一言是国内技术一霸,那自动驾驶中的技术魔王,你听过说吗?另外......
  • 为什么需要代码签名,代码签名是如何工作的?
    进入系统的常见方法之一是通过恶意软件。恶意软件可以到达系统的方式是通过软件。你们中的一些人在安装软件时一定注意到了PC上的未知发布者警告。如果你没有,那么你很幸运,......
  • Redis_IO多路复用底层原理
    从底层了解IO多路复用模型前言当我们去面试的时候,问到了redis,nginx,netty他们的底层模型分别是什么?redis->epollnginx->epollnetty->epoll?需要从操作系统的层......
  • 用于油箱液位检测的电容式液位传感器工作原理
    电容式收位传感器是一种利用电容量的变化来测量液面高低的传感器。一电容量的大小与电容外特征有直接关系,即与极板的面积成正比,与极板间的距离成反比,特别是与极板间的介质......