點擊上方“ 搜雲庫技術團隊 ”關注,選擇“ 設爲星標

回覆“ 1024 ”或 面試題 獲取 4T架構師 資料

什麼是ThreadLocal變量

ThreadLoal 變量,線程局部變量,同一個 ThreadLocal 所包含的對象,在不同的 Thread 中有不同的副本。這裏有幾點需要注意:

  • 因爲每個 Thread 內有自己的實例副本,且該副本只能由當前 Thread 使用。這是也是 ThreadLocal 命名的由來。

  • 既然每個 Thread 有自己的實例副本,且其它 Thread 不可訪問,那就不存在多線程間共享的問題。

ThreadLocal 提供了線程本地的實例。它與普通變量的區別在於,每個使用該變量的線程都會初始化一個完全獨立的實例副本。ThreadLocal 變量通常被private static修飾。當一個線程結束時,它所使用的所有 ThreadLocal 相對的實例副本都可被回收。

總的來說,ThreadLocal 適用於每個線程需要自己獨立的實例且該實例需要在多個方法中被使用,也即變量在線程間隔離而在方法或類間共享的場景。

ThreadLocal實現原理

首先 ThreadLocal 是一個泛型類,保證可以接受任何類型的對象。

因爲一個線程內可以存在多個 ThreadLocal 對象,所以其實是 ThreadLocal 內部維護了一個 Map ,這個 Map 不是直接使用的 HashMap ,而是 ThreadLocal 實現的一個叫做 ThreadLocalMap 的靜態內部類。而我們使用的 get()、set() 方法其實都是調用了這個ThreadLocalMap類對應的 get()、set() 方法。例如下面的 set 方法:

get方法:

createMap方法:

ThreadLocalMap是個靜態的內部類:

最終的變量是放在了當前線程的 ThreadLocalMap 中,並不是存在 ThreadLocal 上,ThreadLocal 可以理解爲只是ThreadLocalMap的封裝,傳遞了變量值。

內存泄漏問題

實際上 ThreadLocalMap 中使用的 key 爲 ThreadLocal 的弱引用,弱引用的特點是,如果這個對象只存在弱引用,那麼在下一次垃圾回收的時候必然會被清理掉。

所以如果 ThreadLocal 沒有被外部強引用的情況下,在垃圾回收的時候會被清理掉的,這樣一來 ThreadLocalMap中使用這個 ThreadLocal 的 key 也會被清理掉。但是,value 是強引用,不會被清理,這樣一來就會出現 key 爲 null 的 value。

ThreadLocalMap實現中已經考慮了這種情況,在調用 set()、get()、remove() 方法的時候,會清理掉 key 爲 null 的記錄。如果說會出現內存泄漏,那只有在出現了 key 爲 null 的記錄後,沒有手動調用 remove() 方法,並且之後也不再調用 get()、set()、remove() 方法的情況下。

使用場景

如上文所述,ThreadLocal 適用於如下兩種場景

  • 每個線程需要有自己單獨的實例

  • 實例需要在多個方法中共享,但不希望被多線程共享

對於第一點,每個線程擁有自己實例,實現它的方式很多。例如可以在線程內部構建一個單獨的實例。ThreadLoca 可以以非常方便的形式滿足該需求。

對於第二點,可以在滿足第一點(每個線程有自己的實例)的條件下,通過方法間引用傳遞的形式實現。ThreadLocal 使得代碼耦合度更低,且實現更優雅。

1)存儲用戶Session

一個簡單的用ThreadLocal來存儲Session的例子:

2)解決線程安全的問題

比如Java7中的SimpleDateFormat不是線程安全的,可以用ThreadLocal來解決這個問題:

這裏的DateUtil.formatDate()就是線程安全的了。(Java8裏的 java.time.format.DateTimeFormatter 是線程安全的,Jodatime裏的DateTimeFormat也是線程安全的)。

參考:

http://www.jasongj.com/java/threadlocal/ https://stackoverflow.com/questions/817856/when-and-how-should-i-use-a-threadlocal-variable https://stackoverflow.com/questions/17968803/threadlocal-memory-leak http://java.jiderhamn.se/2012/01/29/classloader-leaks-iv-threadlocal-dangers-and-why-threadglobal-may-have-been-a-more-appropriate-name/

作者: 阿凡盧

cnblogs.com/luxiaoxun/p/8744826.html

近期技術熱文

1、 IDEA 插件推薦: EasyCode 一鍵幫你生成所需代碼  

2、 IntelliJ IDEA 的 2020 ,真的 很牛皮!(破音)  

3、 大公司爲什麼都有API網關?聊聊API網關的作用  

4、 fastjson這麼快,爲啥老外還是熱衷 jackson?  

5、 SpringBoot+Redis分佈式鎖:模擬搶單場景  

6、 面試官:說下微信掃碼登錄的實現原理?

相關文章