14.Java本地线程(ThreadLocal)

这个在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

你可能感兴趣的:(java,并发)