一张图带你了解ThreadLocal工作原理

ThreadLocal<T> 用于存储线程缓存,用简单的操作实现线程间缓存的操作,做到缓存隔离

下面贴出源码:

//如图第①步 像线程中存入123 至于数据结构怎么存 下面解释
public class App 
{
    public static ThreadLocal<Integer> threadLocal = new ThreadLocal<Integer>();

    public static void main( String[] args )
    {
        threadLocal.set(123);

        threadLocal.get();
    }
}



public void set(T value) {
	//获取当前线程
    Thread t = Thread.currentThread();

    //如图第②步 第③步 获取当前线程t中的threadLocals
    ThreadLocalMap map = getMap(t);

    if (map != null)
    	//如图第④步 向map中put键值对,键为当前threadLocal对象,值为value,
    	//至于map如何操作可近似理解为 HashMap的put操作 不是本文重点不多讲
        map.set(this, value);
    else
    	//如果当前线程threadLocals为空 则创建并以此键值对初始化
        createMap(t, value);
}
//获取线程中的threadLocals
ThreadLocalMap getMap(Thread t) {
    return t.threadLocals;
}
//创建并初始化threadLocals
void createMap(Thread t, T firstValue) {
    t.threadLocals = new ThreadLocalMap(this, firstValue);
}

以上代码简述就是 向Thread中有个threadLocals中存入<ThreadLocal, value> 的键值对,对于其工作原理基本有了大致的了解;

下面介绍线程缓存中的取值代码

贴出取值代码:

public T get() {
    Thread t = Thread.currentThread();
    ThreadLocalMap map = getMap(t);
    if (map != null) {
    	//类似HashMap取值,键为当前threadLocal对象
        ThreadLocalMap.Entry e = map.getEntry(this);
        //如果不为空则做Object的转换
        if (e != null) {
            @SuppressWarnings("unchecked")
            T result = (T)e.value;
            return result;
        }
    }
    //如果map为空则以 键值对<this,null>初始化 Thread t中属性threadLocals
    return setInitialValue();
}
//这段代码就是set的翻版 只不过value的值为null
private T setInitialValue() {
    T value = initialValue();
    Thread t = Thread.currentThread();
    ThreadLocalMap map = getMap(t);
    if (map != null)
        map.set(this, value);
    else
        createMap(t, value);
    return value;
}

总结:ThreadLocal对象通常为全局共享对象,对于每个线程来说,都以此对象为key,

试想如果要从一个map中取值,连Key都不知道,怎么取出预想中的value,实现缓存的存取;

引申思考如果ThreadLocal为局部变量,那么只在当前方法内有效, 出了方法就无法持有该ThreadLocal对象,自然也就无法实现线程内缓存的存取

说到ThreadLocal为局部变量自然就会想到内存泄漏的思考点,有兴趣的同学可以去思考一下

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
ThreadLocal 是 Java 中的一个线程局部变量,它提供了一种在每个线程中存储数据的机制。每个线程都可以独立地访问自己的 ThreadLocal 变量,而不会影响其他线程的访问。 ThreadLocal工作原理是通过为每个线程创建一个独立的副本来实现的。当一个线程访问 ThreadLocal 变量时,它实际上是访问自己的副本。这样就避免了线程安全问题,每个线程都可以拥有自己独立的数据副本。 ThreadLocal 的应用场景包括: 1. 线程上下文信息的传递:在多个方法之间共享某些数据,但又不希望将这些数据作为参数传递。通过将数据存储在 ThreadLocal 中,可以在不传递参数的情况下,在不同方法之间共享数据。 2. 数据库连接和事务管理:在使用数据库连接池时,可以将每个线程的数据库连接存储在 ThreadLocal 中,确保每个线程使用自己的数据库连接,避免线程间的干扰。 3. 线程安全的日期格式化:日期格式化类通常不是线程安全的,使用 ThreadLocal 可以为每个线程创建一个独立的日期格式化对象,避免多线程并发访问时的线程安全问题。 4. 线程级别的缓存:在多线程环境下,可以使用 ThreadLocal 实现线程级别的缓存,每个线程都有自己独立的缓存,避免了线程间的数据竞争问题。 5. Web 应用中的用户身份管理:在 Web 应用中,可以使用 ThreadLocal 存储当前用户的信息,方便在不同层之间获取用户身份信息,如用户认证、权限控制等。 这些应用场景都是为了解多线程环境下的线程安全问题,通过使用 ThreadLocal 可以在每个线程中存储独立的数据,避免了线程间的数据竞争和并发访问的问题。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值