【Java ThreadLocal的使用】

原文鏈接  作者: Jakob Jenkov 譯者: 林威建[[email protected]]

Java中的ThreadLocal類允許我們創建只能被同一個線程讀寫的變量。因此,如果一段代碼含有一個ThreadLocal變量的引用,即使兩個線程同時執行這段代碼,它們也無法訪問到對方的ThreadLocal變量。

如何創建ThreadLocal變量

以下代碼展示瞭如何創建一個ThreadLocal變量:

[size=1em][size=1em]

1

private ThreadLocal myThreadLocal = new ThreadLocal();




我們可以看到,通過這段代碼實例化了一個ThreadLocal對象。我們只需要實例化對象一次,並且也不需要知道它是被哪個線程實例化。雖然所有的線程都能訪問到這個ThreadLocal實例,但是每個線程卻只能訪問到自己通過調用ThreadLocal的set()方法設置的值。即使是兩個不同的線程在同一個ThreadLocal對象上設置了不同的值,他們仍然無法訪問到對方的值。

如何訪問ThreadLocal變量

一旦創建了一個ThreadLocal變量,你可以通過如下代碼設置某個需要保存的值:

[size=1em][size=1em]

1

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




可以通過下面方法讀取保存在ThreadLocal變量中的值:

[size=1em][size=1em]

1

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




get()方法返回一個Object對象,set()對象需要傳入一個Object類型的參數。

爲ThreadLocal指定泛型類型

我們可以創建一個指定泛型類型的ThreadLocal對象,這樣我們就不需要每次對使用get()方法返回的值作強制類型轉換了。下面展示了指定泛型類型的ThreadLocal例子:

[size=1em][size=1em]

1

private ThreadLocal myThreadLocal = newThreadLocal<String>();




現在我們只能往ThreadLocal對象中存入String類型的值了。

並且我們從ThreadLocal中獲取值的時候也不需要強制類型轉換了。

如何初始化ThreadLocal變量的值

由於在ThreadLocal對象中設置的值只能被設置這個值的線程訪問到,線程無法在ThreadLocal對象上使用set()方法保存一個初始值,並且這個初始值能被所有線程訪問到。

但是我們可以通過創建一個ThreadLocal的子類並且重寫initialValue()方法,來爲一個ThreadLocal對象指定一個初始值。就像下面代碼展示的那樣:

  1. private ThreadLocal myThreadLocal = new ThreadLocal<String>() {


  2.     @Override


  3.     protected String initialValue() {


  4.         return "This is the initial value";


  5.     }

  6. };


複製代碼

一個完整的ThreadLocal例子

下面是一個完整的可執行的ThreadLocal例子:


  1. public class ThreadLocalExample {

  2. public static class MyRunnable implements Runnable {


  3.         private ThreadLocal threadLocal = new ThreadLocal();


  4. @Override

  5. public void run() {


  6.             threadLocal.set((int) (Math.random() * 100D));

  7. try {


  8.             Thread.sleep(2000);


  9.             } catch (InterruptedException e) {


  10. }


  11.             System.out.println(threadLocal.get());


  12.         }


  13.     }


  14.     public static void main(String[] args) {


  15.          MyRunnable sharedRunnableInstance = new MyRunnable();


  16.          Thread thread1 = new Thread(sharedRunnableInstance);


  17.          Thread thread2 = new Thread(sharedRunnableInstance);


  18.          thread1.start();


  19.          thread2.start();


  20.     }

  21. }


複製代碼

面的例子創建了一個MyRunnable實例,並將該實例作爲參數傳遞給兩個線程。兩個線程分別執行run()方法,並且都在ThreadLocal實例上保存了不同的值。如果它們訪問的不是ThreadLocal對象並且調用的set()方法被同步了,則第二個線程會覆蓋掉第一個線程設置的值。但是,由於它們訪問的是一個ThreadLocal對象,因此這兩個線程都無法看到對方保存的值。也就是說,它們存取的是兩個不同的值。

關於InheritableThreadLocal

InheritableThreadLocal類是ThreadLocal類的子類。ThreadLocal中每個線程擁有它自己的值,與ThreadLocal不同的是,InheritableThreadLocal允許一個線程以及該線程創建的所有子線程都可以訪問它保存的值。

【注:所有子線程都會繼承父線程保存的ThreadLocal值】

原創文章,轉載請註明: 轉載自ifeve.com


發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章