Java ThreadLocal 应用

一、需求介绍
由于项目需要在同一进程不同方法内获取到上下文内容,而在同一线程父方法可以获取到上下文内容,子方法由不同的项目组提供和开发,然后以jar包的方式打包,这时候问题就出现了,父方法的上下文内容如何可以提供给子方法,并在子方法中获取到对应线程的上下文内容?

注:每个线程的上下文内容是不同的,父类只提供集成容器,具体方法的实现由不同的项目组和部门编写

二、解决思路
应用Java 提供的ThreadLocal

首先查看下TheadLocal提供的方法
T get()
          返回此线程局部变量的当前线程副本中的值。
protected  T initialValue()
          返回此线程局部变量的当前线程的初始值。
void remove()
          移除此线程局部变量的值。
void set(T value)
          将此线程局部变量的当前线程副本中的值设置为指定值。


2.1、get 方法:
public T get() {
    Thread t = Thread.currentThread();
    ThreadLocalMap map = getMap(t);
    if (map != null) {
        ThreadLocalMap.Entry e = map.getEntry(this);
        if (e != null)
            return (T)e.value;
     }
    return setInitialValue();
}

说明:
1、首先获取当前线程
2、获取当前线程的ThreadLocalMap即threadLocals
getMap(Thread t)源码为:
ThreadLocalMap getMap(Thread t) {
    return t.threadLocals;
}
3、如果当前线程存在threadLocals则根据当前的ThreadLocal获取threadLocals的值.(threadLocals其实就是一Map)
4、如果当前线程的ThreadLocalMap为null,则执行方法setInitialValue()

可以看的出是获取Thread的threadLocals
Thread中源码为如下:
ThreadLocal.ThreadLocalMap threadLocals = null;

2.2、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);
}
源码说明:
1、首先获取当前线程
2、获取当前线程ThreadLocalMap
3、如果ThreadLocalMap不为null set(ThreadLocal,value)
4、如果为null createMap(Thread,value);

备注:这里可以看出不为null时,set的内容为key:ThreadLocal,value:value,也就是说每个线程都有一个ThreadLocalMap这个map内存放得内容为ThreadLocal,value

接下来在看下createMap源码
void createMap(Thread t, T firstValue) {
    t.threadLocals = new ThreadLocalMap(this, firstValue);
}

备注:该方法为当前线程的内部变量threadLocals进行了赋值操作,而ThreadLocalMap为ThreadLocal的内部类。new ThreadLocalMap(this, firstValue) 内部进行的操作也是为当前线程的threadLocals变量进行了set操作,key为当前的ThreadLocal,value为传入的firstValue

2.3、setInitialValue方法

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;
}

该方法返回null,并会创建当前线程的threadLocals

综上:应用ThreadLocal在线程中进行上下文同步时,要保持一致的ThreadLocal。当多线程应用线程池时,要保证线程池中相同的线程每次拿到得ThreadLocal要不相同,否则会出现问题。


回到问题的开始,解决该问题的方案为
1、创建一个单例类(ex:ThreadLocalTest)保证线程上下文拿到得ThreadLocal都一致
2、父类提供给子类可以获取单例类ThreadLocalTest的方法。(要是子类分别由不同的部门、项目组以jar的方式提供时,负责父类服务的可以给项目组提供一个通用的jar文件)
3、在调用父类多线程入口时获取ThreadLocalTest单例,set(上下文内容)
4、子类在方法内获取ThreadLocalTest单例调用get()方法获取当前线程的value即为上下文内容
5、父类调用子方法后执行ThreadLocalTest.remove()操作

public class ThreadLocalTest<T> {
    private static ThreadLocal local = new ThreadLocal();
    private static ThreadLocalTest threadlocaltest = null;

    public synchronized ThreadLocalTest getInitialize(){
        if(threadlocaltest == null){
            threadlocaltest = new ThreadLocalTest<T>();
        }
        return threadlocaltest;
    }

    public T get(){
        return (T) local.get();
    }

    public void set(T t){
        local.set(t);
    }

    public void remove(){
        local.remove();
    }
}



后记:
很多人问我,在父类中既然可以获取到上下文那直接传给子类方法就行了啊。注意前提说的是父类只负责提供一个容器,具体的实现由不同的部门和项目组编写

猜你喜欢

转载自polim.iteye.com/blog/1254447