還不會使用Java ThreadLocal落後了吧!

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

如何創建ThreadLocal變數

以下程式碼展示了如何創建一個ThreadLocal變數:

private ThreadLocal myThreadLocal = new ThreadLocal();

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

如何訪問ThreadLocal變數

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

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

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

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

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

為ThreadLocal指定泛型類型

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

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

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

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

如何初始化ThreadLocal變數的值

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

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

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

    @Override
    protected String initialValue() {
        return "This is the initial value";
    }

};

一個完整的ThreadLocal例子

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

public class ThreadLocalExample {

    public static class MyRunnable implements Runnable {

        private ThreadLocal threadLocal = new ThreadLocal();

        @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();
    }

}

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

關於InheritableThreadLocal

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

最後

私信回復 資料 領取一線大廠Java面試題總結+阿里巴巴泰山手冊+各知識點學習思維導+一份300頁pdf文檔的Java核心知識點總結!

這些資料的內容都是面試時面試官必問的知識點,篇章包括了很多知識點,其中包括了有基礎知識、Java集合、JVM、多執行緒並發、spring原理、微服務、Netty 與RPC 、Kafka、日記、設計模式、Java演算法、資料庫、Zookeeper、分散式快取、數據結構等等。
file

Tags: