这个在Java中的ThreadLocal类可以保证使你创建的变量只能被相同的线程读和写。因此,甚至如果两个线程正在执行相同的代码,并且这个代码有一个对于ThreadLocal变量的引用,然后这两个线程就不能看到彼此的ThreadLocal变量。
创建一个ThreadLocal
这里有一个代码现实如何创建一个ThreadLocal:
private ThreadLocal myThreadLocal = new ThreadLocal();
访问一个ThreadLocal
一旦一个ThreadLocal被创建了,你就可以像下面这样设置一个值去存储:
myThreadLocal.set("A thread local value");
String threadLocalValue = (String) myThreadLocal.get();
泛型化的ThreadLocal
你可以创建一个泛型化的ThreadLocal,以至于你不用在调用get方法的时候进行强制转化了。这里有一个例子:
private ThreadLocal<String> myThreadLocal = new ThreadLocal<String>();
现在你只能存储存储字符串类型在ThreadLocal实例中。另外的,你不需要强制转化这个值了:
myThreadLocal.set("Hello ThreadLocal"); String threadLocalValue = myThreadLocal.get();
因为设置一个ThreadLocal对象的值只是对于设置这个值的线程可见的,所以没有线程可以使用set方法设置ThreadLocal的值对所有的线程可见。
代替的,你可以通过子类ThreadLocal特指一个初始化的值对于一个ThreadLocal对象,以及覆盖initialValue方法。像下面这样:
private ThreadLocal myThreadLocal = new ThreadLocal<String>() { @Override protected String initialValue() { return "This is the initial value"; } };
完整的ThreadLocal实例
这里有一个完整运行的ThreadLocal实例
public class ThreadLocalExample { public static class MyRunnable implements Runnable { private ThreadLocal<Integer> threadLocal = new ThreadLocal<Integer>(); @Override public void run() { threadLocal.set( (int) (Math.random() * 100D) ); try { Thread.sleep(2000); } catch (InterruptedException e) { } System.out.println(threadLocal.get()); } } public static void main(String[] args) { MyRunnable sharedRunnableInstance = new MyRunnable(); Thread thread1 = new Thread(sharedRunnableInstance); Thread thread2 = new Thread(sharedRunnableInstance); thread1.start(); thread2.start(); thread1.join(); //wait for thread 1 to terminate thread2.join(); //wait for thread 2 to terminate } }
这个例子创建了一个单独的MyRunnable实例,它被传递给了两个不同的线程。这两个线程都执行了run方法,并且在ThreadLocal实例上设置了不同的值。如果对于这个set方法的调用访问是同步的,并且它还没有使用ThreadLocal对象,第二个线程将会覆盖第一个线程设置的值。
然而,因为它是一个ThreadLocal对象,不能看到彼此的值。因此,他们设置和得到不同的值。
InheritableThreadLocal
这个InheritableThreadLocal类是ThreadLocal类的子类。代替的每一个线程在一个ThreadLoca内部都有它自己的值,这个类同意访问对于一个线程的值,并且被那个线程创建的所有子线程。
翻译地址:http://tutorials.jenkov.com/java-concurrency/threadlocal.html