有三个系统A,B,C;A,B需要通过调用C的接口c1,C在接口中针对A,B传给接口的companyCode会拿到公司companyId放入到ThreadLocal,companyId作为mybatis拦截器中参数会在所有的sql操作的where条件中加上,假如C有个c2接口也会使用ThreadLocal变量,这是应该获取不到这个变量,但实际下来c2接口也会拿到值,这是为什么?
通知一个请求会是一个线程,也就说A,B调用c1会各自线程处理,c2被调用也会有自己的线程处理,ThreadLocal按理说应该取不到,因为ThreadLocal以当前线程做为Map key来存储和使用对象的,源码实现如下
public void set(T value) { Thread t = Thread.currentThread(); ThreadLocalMap map = getMap(t); if (map != null) map.set(this, value); else createMap(t, value);}
public T get() { Thread t = Thread.currentThread(); ThreadLocalMap map = getMap(t); if (map != null) { ThreadLocalMap.Entry e = map.getEntry(this); if (e != null) { @SuppressWarnings("unchecked") T result = (T)e.value; return result; } } return setInitialValue();}
取到了说明c2和c1接口共用了一个线程,这与一个请求生成一个线程相悖,后来查可能与tomcat线程池有关系,也就是c2与c1使用了tomcat线程池的同一个线程。