ThreadLocal使用及原理

ThreadLocal使用及原理

ThreadLocal类主要解决的就是每个线程绑定自己的值,可以将ThreadLocal类比喻成全局存放数据的盒子,盒子中可以存储每个线程的私有数据。

1.使用demo

比较简单的存取

public class ThreadLocalTest1 {
    public static void main(String[] args) {
        ThreadLocal<Integer> threadLocal = new ThreadLocal<>();
        ThreadLocal<Integer> threadLocal1 = new ThreadLocal<>();
        threadLocal.set(1);
        threadLocal1.set(2);
        System.out.println("threadLocal=" + threadLocal.get());
        System.out.println("threadLocal1=" + threadLocal1.get());
    }
}

运行结果

threadLocal=1
threadLocal1=2

2.实战

当前用户。绑定当前登录的用户到http request线程,在拦截器赋值,在业务代码中获取使用。

只要不新开一个线程,那么在controller和service层就可以直接MingSanContext.getCurrentUser()获取到当前登录的用户信息。

如果想在子线程也能获取到,在项目源码中获取方案,项目地址在文章末尾。

注意:一定要remove,不然会内存泄漏,甚至溢出。

public class MingSanContext {
    private static final ThreadLocal<CurrentUser> LOGIN_THREAD_LOCAL = new ThreadLocal<>();
    
    public static void setCurrentUser(CurrentUser currentUser) {
        if (currentUser != null) {
            LOGIN_THREAD_LOCAL.set(currentUser);
        }
    }

    /**
     * 获取当前访问用户
     * <p>
     * 返回CurrentUser这个对象理论上不会为空,里面的参数可能为null,
     * <p>
     * 但是网关也有拦截
     * <p>
     * 建议 CurrentUser可以不判断,currentUser的参数要判断
     *
     * @return 用户信息
     */
    public static CurrentUser getCurrentUser() {
        return LOGIN_THREAD_LOCAL.get();
    }

    /**
     * 清除当前用户信息
     */
    public static void removeCurrentUser() {
        LOGIN_THREAD_LOCAL.remove();
    }
}

3.原理

一切皆在Thread类中。

Thread类包含两个ThreadLocal.ThreadLocalMap。默认的构造方法,会初始化inheritableThreadLocals,不会初始化threadLocals

public class Thread implements Runnable {
    /* ThreadLocal values pertaining to this thread. This map is maintained
         * by the ThreadLocal class. */
    ThreadLocal.ThreadLocalMap threadLocals = null;

    /*
         * InheritableThreadLocal values pertaining to this thread. This map is
         * maintained by the InheritableThreadLocal class.
         */
    ThreadLocal.ThreadLocalMap inheritableThreadLocals = null;
}

再看ThreadLocal类的set方法,获取当前线程的ThreadLocalMap Thread.threadLocals。第一次获取肯定是null,因为new Thread()不会初始化threadLocals,调用createMap方法,给当前线程的threadLocals赋值。ThreadLocalMap的key是当前ThreadLocal对象,值是调用方set的值。

public class ThreadLocal<T> {
	public void set(T value) {
        Thread t = Thread.currentThread();
        ThreadLocalMap map = getMap(t);
        if (map != null) {
            map.set(this, value);
        } else {
            createMap(t, value);
        }
    }
    ThreadLocalMap getMap(Thread t) {
        return t.threadLocals;
    }
    void createMap(Thread t, T firstValue) {
        t.threadLocals = new ThreadLocalMap(this, firstValue);
    }
}

4.注意

ThreadLocal只可以在当前线程中使用,子线程中获取不到值。

public class ThreadLocalTest2 {
    public static void main(String[] args) {
        ThreadLocal<Integer> threadLocal = new ThreadLocal<>();
        threadLocal.set(1);
        System.out.println("当前线程获取ThreadLocal值:" + threadLocal.get());
        new Thread(new Runnable() {
            @Override
            public void run() {
                System.out.println("子线程获取ThreadLocal值:" + threadLocal.get());
            }
        }).start();
    }
}

运行结果

当前线程获取ThreadLocal:1
子线程获取ThreadLocal:null

ThreadLocal实战,源码地址:highway-resources: threadlocal-current-user

关注公众号- 编程highway - 获取更多好文和学习资料

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

编程还未

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值