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" ; |
此時,在set()方法調用前,當調用get()方法的時候,所有線程都可以看到同一個初始化值。
5、Full ThreadLocal Example
以下是一個完整的ThreadLocal示例:
01 |
public class ThreadLocalExample
{ |
03 |
public static class MyRunnable implements Runnable
{ |
05 |
private ThreadLocal<Integer>
threadLocal = |
06 |
new ThreadLocal<Integer>(); |
10 |
threadLocal.set(
( int )
(Math.random() * 100D) ); |
14 |
} catch (InterruptedException
e) { |
17 |
System.out.println(threadLocal.get()); |
21 |
public static void main(String[]
args) { |
22 |
MyRunnable
sharedRunnableInstance = new MyRunnable(); |
24 |
Thread
thread1 = new Thread(sharedRunnableInstance); |
25 |
Thread
thread2 = new Thread(sharedRunnableInstance); |
上面創建了兩個線程共享一個MyRunnable實例。每個線程執行run()方法的時候,會給同一個ThreadLocal實例設置不同的值。如果調用set()方法的時候用synchronized關鍵字同步,而且不是一個ThreadLocal對象實例,那麼第二個線程將會覆蓋第一個線程所設置的值。
然而,由於是ThreadLocal對象,所以兩個線程無法看到彼此的值。因此,可以設置或者獲取不同的值。
6、InheritableThreadLocal
InheritableThreadLocal類是ThreadLocal的子類。爲了解決ThreadLocal實例內部每個線程都只能看到自己的私有值,所以InheritableThreadLocal允許一個線程創建的所有子線程訪問其父線程的值。