14.Java本地线程(ThreadLocal)

来源:互联网 发布:心理学书籍推荐 知乎 编辑:程序博客网 时间:2024/05/29 04:37

这个在Java中的ThreadLocal类可以保证使你创建的变量只能被相同的线程读和写。因此,甚至如果两个线程正在执行相同的代码,并且这个代码有一个对于ThreadLocal变量的引用,然后这两个线程就不能看到彼此的ThreadLocal变量。

创建一个ThreadLocal

这里有一个代码现实如何创建一个ThreadLocal:

private ThreadLocal myThreadLocal = new ThreadLocal();

正如你看到的,你实例化一个新的ThreadLocal对象。这个每个线程只需要做一次。甚至如果不同的线程执行访问一个ThreadLocal的相同的代码,每一个线程将只会看到它自己的ThreadLocal实例。甚至如果两个线程设置不同的值在这相同的ThreadLocal对象上,它们都不会看到彼此的值。

访问一个ThreadLocal

一旦一个ThreadLocal被创建了,你就可以像下面这样设置一个值去存储:

myThreadLocal.set("A thread local value");

你可以像下面这样去读这个值:

String threadLocalValue = (String) myThreadLocal.get();

这个get方法返回一个对象,并且这个set方法传递一个对象作为参数。

泛型化的ThreadLocal

你可以创建一个泛型化的ThreadLocal,以至于你不用在调用get方法的时候进行强制转化了。这里有一个例子:

private ThreadLocal<String> myThreadLocal = new ThreadLocal<String>();

现在你只能存储存储字符串类型在ThreadLocal实例中。另外的,你不需要强制转化这个值了:

myThreadLocal.set("Hello ThreadLocal");String threadLocalValue = myThreadLocal.get();

初始化ThreadLocal值

因为设置一个ThreadLocal对象的值只是对于设置这个值的线程可见的,所以没有线程可以使用set方法设置ThreadLocal的值对所有的线程可见。

代替的,你可以通过子类ThreadLocal特指一个初始化的值对于一个ThreadLocal对象,以及覆盖initialValue方法。像下面这样:

private ThreadLocal myThreadLocal = new ThreadLocal<String>() {    @Override protected String initialValue() {        return "This is the initial value";    }};    

现在所有的线程可以看到相同的初始化值,在调用set方法之前。

完整的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

0 0