閱讀548 返回首頁    go 阿裏雲 go 技術社區[雲棲]


《Java並發性和多線程介紹》-Java TheadLocal

Java中的ThreadLocal類可以讓你創建的變量隻被同一個線程進行讀和寫操作。因此,盡管有兩個線程同時執行一段相同的代碼,而且這段代碼又有一個指向同一個ThreadLocal變量的引用,但是這兩個線程依然不能看到彼此的ThreadLocal變量域。

1、創建一個ThreadLocal對象

2、訪問ThreadLocal對象

3、ThreadLocal泛型

4、初始化ThreadLocal

5、Full ThreadLocal Example

6、InheritableThreadLocal

1、創建一個ThreadLocal對象

如下所示,創建一個ThreadLocal變量:

1 private ThreadLocal myThreadLocal = new ThreadLocal();

你實例化了一個ThreadLocal對象。每個線程僅需要實例化一次即可。雖然不同的線程執行同一段代碼時,訪問同一個ThreadLocal變量,但是每個線程隻能看到私有的ThreadLocal實例。所以不同的線程在給ThreadLocal對象設置不同的值時,他們也不能看到彼此的修改。

2、訪問ThreadLocal對象

一旦創建了一個ThreadLocal對象,你就可以通過以下方式來存儲此對象的值:

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

也可以直接讀取一個ThreadLocal對象的值:

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

get()方法會返回一個Object對象,而set()方法則依賴一個Object對象參數。

3、ThreadLocal泛型

為了使get()方法返回值不用做強製類型轉換,通常可以創建一個泛型化的ThreadLocal對象。以下就是一個泛型化的ThreadLocal示例:

1 private ThreadLocal myThreadLocal1 = new ThreadLocal<String>();

現在你可以存儲一個字符串到ThreadLocal實例裏,此外,當你從此ThreadLocal實例中獲取值的時候,就不必要做強製類型轉換。

1 myThreadLocal1.set("Hello ThreadLocal");
2 String threadLocalValues = myThreadLocal.get();

4、初始化ThreadLocal

由於ThreadLocal對象的set()方法設置的值隻對當前線程可見,那有什麼方法可以為ThreadLocal對象設置的值對所有線程都可見。

為此,我們可以通過ThreadLocal子類的實現,並覆寫initialValue()方法,就可以為ThreadLocal對象指定一個初始化值。如下所示:

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

此時,在set()方法調用前,當調用get()方法的時候,所有線程都可以看到同一個初始化值。

5、Full ThreadLocal Example

以下是一個完整的ThreadLocal示例:

01 public class ThreadLocalExample {
02  
03     public static class MyRunnable implements Runnable {
04  
05         private ThreadLocal<Integer> threadLocal =
06                new ThreadLocal<Integer>();
07  
08         @Override
09         public void run() {
10             threadLocal.set( (int) (Math.random() * 100D) );
11  
12             try {
13                 Thread.sleep(2000);
14             catch (InterruptedException e) {
15             }
16  
17             System.out.println(threadLocal.get());
18         }
19     }
20  
21     public static void main(String[] args) {
22         MyRunnable sharedRunnableInstance = new MyRunnable();
23  
24         Thread thread1 = new Thread(sharedRunnableInstance);
25         Thread thread2 = new Thread(sharedRunnableInstance);
26  
27         thread1.start();
28         thread2.start();
29  
30         thread1.join(); //wait for thread 1 to terminate
31         thread2.join(); //wait for thread 2 to terminate
32     }
33  
34 }

上麵創建了兩個線程共享一個MyRunnable實例。每個線程執行run()方法的時候,會給同一個ThreadLocal實例設置不同的值。如果調用set()方法的時候用synchronized關鍵字同步,而且不是一個ThreadLocal對象實例,那麼第二個線程將會覆蓋第一個線程所設置的值。

然而,由於是ThreadLocal對象,所以兩個線程無法看到彼此的值。因此,可以設置或者獲取不同的值。

6、InheritableThreadLocal

InheritableThreadLocal類是ThreadLocal的子類。為了解決ThreadLocal實例內部每個線程都隻能看到自己的私有值,所以InheritableThreadLocal允許一個線程創建的所有子線程訪問其父線程的值。

最後更新:2017-05-22 11:01:49

  上一篇:go  《雲數據管理:挑戰與機遇》導讀
  下一篇:go  《雲數據管理:挑戰與機遇》一2.3 數據庫係統