springboot 多数据源二 之 ThreadLocal维护数据源类

一、ThreadLocal简单介绍

首先,ThreadLocal是用来维护本线程的变量的,并不能解决共享变量的并发问题。ThreadLocal是各线程将值存入该线程的map中,以ThreadLocal自身作为key,需要用时获得的是该线程之前存入的值。如果存入的是共享变量,那取出的也是共享变量,并发问题还是存在的。

简单看一下例子:

 
public class TestThreadLocal {

private static final ThreadLocal<String> threadLocalA = new ThreadLocal<>();

private static final ThreadLocal<String> threadLocalB = new ThreadLocal<>();


/**

* 在调用的线程的map中存入key为ThreadLocal本身,value为在该线程设置的值

* @param value

*/

public static void setValueA(String value){

threadLocalA.set(value);

}


public static String getValueA(){

return threadLocalA.get();

}


public static void clearValueA(){

threadLocalA.remove();

}


public static void setValueB(String value){

threadLocalB.set(value);

}


public static String getValueB(){

return threadLocalB.get();

}


public static void clearValueB(){

threadLocalB.remove();

}



public static void main(String[] args) {

//线程1的ThreadLocalMap中存着key为threadLocalA,value为A1;key为threadLocalB,value为B1

new Thread(){

@Override

public void run(){

setValueA("A1");

System.out.println("thread1:" + getValueA());

clearValueA();


setValueB("B1");

System.out.println("thread1:" + getValueB());

clearValueB();

}

}.start();


//线程2的ThreadLocalMap中存着key为threadLocalA,value为A2;key为threadLocalB,value为B2

new Thread(){

@Override

public void run(){

setValueA("A2");

System.out.println("thread2:" + getValueA());

clearValueA();


setValueB("B2");

System.out.println("thread2:" + getValueB());

clearValueB();

}

}.start();

}

}

该例子的执行结果为:

 
thread2:A2

thread2:B2

thread1:A1

thread1:B1

从该例子可以看出多个线程设置ThreadLocal的值只在该线程的作用范围内有效。操作ThreadLocal的set,get方法实际上是操作该线程的一个代理,其本质是在该线程的ThreadLocalMap中存储了key为ThreadLocal本身和对应的值。

以下是ThreadLocal中的set方法:

 
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;

}

 

要使用ThreadLocal维护数据源类 的原因:

,如果将key作为静态变量那可能引起并发问题,当同时访问数据库时,一个线程刚刚设置的key可能被另一个线程修改了,导致最终访问的数据源不正确。那么怎么样才能保证key能不被其它线程修改呢,即不能控制并发也不能每个线程都实例化DynamicDataSource来设置该线程的key,这时候ThreadLocal就能起到很好的作用,保护该线程私有的变量。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值