这个在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 myThreadLocal = new ThreadLocal();
现在你只能存储存储字符串类型在ThreadLocal实例中。另外的,你不需要强制转化这个值了:
myThreadLocal.set("Hello ThreadLocal");
String threadLocalValue = myThreadLocal.get();
因为设置一个ThreadLocal对象的值只是对于设置这个值的线程可见的,所以没有线程可以使用set方法设置ThreadLocal的值对所有的线程可见。
代替的,你可以通过子类ThreadLocal特指一个初始化的值对于一个ThreadLocal对象,以及覆盖initialValue方法。像下面这样:
private ThreadLocal myThreadLocal = new ThreadLocal() {
@Override protected String initialValue() {
return "This is the initial value";
}
};
完整的ThreadLocal实例
这里有一个完整运行的ThreadLocal实例
public class ThreadLocalExample {
public static class MyRunnable implements Runnable {
private ThreadLocal threadLocal =
new ThreadLocal();
@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