ThreadLocal概念和源码讲解

ThreadLocal概念

ThreadLocal<T> 提供了线程的局部变量,每个线程都可以通过set(),get() 来对这个局部变量进行操作,但不会和其他线程的局部变量冲突,实现了线程的数据隔离。也就是说线程操作了这个ThreadLocal<T>变量那么它就独自拥有这个操作的值,其他线程访问不了。ThreadLocal<T>的泛型变量可以封装各种类型,是线程封装的线程变量。

ThreadLocal内部封装了ThreadLocalMap内部类,而Thread类内部聚合了一个ThreadLocalMap:ThreadLocalMap内部封装了一个Entry类用于存储数据(如下图),可以看出Entry继承了一个虚引用,Entry的存储结构是键值对,而这个键就是一个ThreadLocal对象,值就是我们需要存储的线程私有变量。

static class ThreadLocalMap {

       
        static class Entry extends WeakReference<ThreadLocal<?>> {
            /** The value associated with this ThreadLocal. */
            Object value;

            Entry(ThreadLocal<?> k, Object v) {
                super(k);
                value = v;
            }
        }

 

从ThreadLocal获取值

public T get() {
        Thread t = Thread.currentThread();  //获取当前线程
        ThreadLocalMap map = getMap(t);  //获取线程内聚合的ThreadLocalMap
        if (map != null) {
            ThreadLocalMap.Entry e = map.getEntry(this);  //通过ThreadLocal对象获取Entry
            if (e != null) {
                @SuppressWarnings("unchecked")
                T result = (T)e.value;  //获取值
                return result;
            }
        }
        return setInitialValue();  //若map为null,初始化
    }

可以看到,首先通过Thread.currentThread()获取当前线程,然后再从线程里获取一个ThreadLocalMap,每个线程都会维护一个ThreadLocalMap的引用;然后再从该map里获取值,key是一个ThreadLocal对象。

往ThreadLoacal中set值

public void set(T value) {
        Thread t = Thread.currentThread(); //获取当前线程
        ThreadLocalMap map = getMap(t);  //获取当前线程关联的map
        if (map != null)
            map.set(this, value);  //往map插入值,key是一个ThreadLocal对象
        else
            createMap(t, value);  //若map为null,新建一个当前对象的ThreadLocalMap并存入value
    }

 

在spring中应用场景

在Spring的获取数据库连接池的连接用到,即当线程需要操作mapper时,mapper封装了sqlsession, sqlsession需要获取连接collection,基于连接获取statement才能对数据库进行操作;线程获取连接的时候不是直接去连接池随机获取一个连接,这样会导致spring事务乱套,因为事务必须基于同一个连接,如果两个操作组成一个事务同时去连接池获取两个不同的连接,那么两个操作就是相互隔离的无法组成事务。

所以获取连接是通过一个事务管理器,事务管理器去连接池取一个,那么问题来了,当不同的线程事务取连接可能取到相同的连接,这样又会导致事务乱套,这时候就需要用到ThreadLocal<Collection>了,每一个线程来取的时候就将这个连接通过ThreadLocal内置到自己线程中私有了,而其他线程是无法得到的,当其他线程来的时候发现自己的ThreadLocal<Collection>为null则又会重新从连接池获取一个连接给自己私有,spring通过这样控制事务的一致性。如下图:

 

 

 

 

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
ThreadLocal码是Java中一个关键的类,它提供了一种在多线程环境下实现线程本地变量的机制。在JDK 8之前和之后,ThreadLocal的内部结构有所变化。ThreadLocal码分为两部分:ThreadLocal类和ThreadLocalMap类。 ThreadLocal类是一个泛型类,它包含了两个核心方法:set()和get()。set()方法用于将一个值与当前线程关联起来,get()方法用于获取当前线程关联的值。 ThreadLocalMap类是ThreadLocal的内部类,它用于存储每个线程的本地变量。在JDK 8之前,ThreadLocalMap是通过线性探测法解决哈希冲突的,每个ThreadLocal对象都对应一个Entry对象,Entry对象包含了ThreadLocal对象和与之关联的值[2]。 在JDK 8之后,ThreadLocalMap的实现方式发生了改变。使用了类似于HashMap的方式,采用了分段锁的机制来提高并发性能。每个线程维护一个ThreadLocalMap对象,其中的Entry对象也是采用链表的形式来解决哈希冲突。 总结起来,ThreadLocal码主要由ThreadLocal类和ThreadLocalMap类组成。ThreadLocal类提供了set()和get()方法来管理线程本地变量,而ThreadLocalMap类则负责存储每个线程的本地变量,并解决哈希冲突的问题。 史上最全ThreadLocal 详解 ThreadLocal码分析_02 内核(ThreadLocalMap) 【JDK码】线程系列之ThreadLocal 深挖ThreadLocal ThreadLocal原理及内存泄露预防 ThreadLocal原理详解——终于弄明白了ThreadLocal ThreadLocal使用与原理 史上最全ThreadLocal 详解。 ThreadLocal码分析,主要有ThreadLocal码以及ThreadLocal的内部结构在jdk8前后的变化。 使用方式非常简单,核心就两个方法set/get public class TestThreadLocal { private static final ThreadLocal<String> threadLocal = new ThreadLocal<>(); public static void main(String[] args) { new Thread(new Runnable() { @Override public void run() { try { threadLocal.set("aaa"); Thread.sleep(500); System.out.println("threadA:" threadLocal.get()); } catch (InterruptedException e) { e.printStackTrace(); } } }).start(); new Thread(new Runnable() { @Override public void run() { threadLocal.set("bbb"); System.out.println("threadB:" threadLocal.get()); } }).start(); } }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值