ThreadLocal工作原理及用法案例
ThreadLocal是什么
ThreadLocal是線程Thread中屬性threadLocals即ThreadLocal.ThreadLocalMap的管理者,ThreadLocal用于給每個(gè)線程操作自己線程的本地變量,通過線程私有從而保證線程安全性。
ThreadLocal原理
拿get()
方法來說,線程的本地變量是存放在線程實(shí)例的屬性ThreadLocalMap上的,ThreadLocalMap本質(zhì)上就是一個(gè)HashMap,ThreadLocal只是一個(gè)管理者,當(dāng)我們的線程需要拿到自己的本地變量時(shí),我們直接調(diào)用ThreadLocal去get本地變量即可。
因?yàn)?code>get()方法底層會(huì)先獲取到當(dāng)前線程,然后通過當(dāng)前線程拿到他的屬性值ThreadLocalMap,如果ThreadLocalMap為空,則會(huì)調(diào)用ThreadLocal的初始化方法拿到初始值返回,如果不為空,則會(huì)拿該ThreadLocal作為key去獲取該線程下的ThreadLocalMap里對(duì)應(yīng)的value值。
ThreadLocal內(nèi)存泄漏問題
線程的屬性值ThreadLocalMap中使用的 key 為 ThreadLocal 的弱引用,而value是強(qiáng)引用。所以,如果ThreadLocal沒有被外部強(qiáng)引用的情況下,在垃圾回收的時(shí)候,key 會(huì)被清理掉,而value 不會(huì)被清理掉。這樣的話,ThreadLocalMap 中就會(huì)出現(xiàn) key 為 null 的 Entry。假如我們不做任何措施的話,value 永遠(yuǎn)無法被 GC 回收,這個(gè)時(shí)候就可能會(huì)產(chǎn)生內(nèi)存泄露。
因此針對(duì)這種情況,我們有兩種原則:
- ThreadLocal申明為private static final。JDK建議ThreadLocal定義為private static,這樣ThreadLocal的弱引用問題則不存在了。
- private與final 盡可能不讓他人修改變更引用。
- static 表示為類屬性,只有在程序結(jié)束才會(huì)被回收。
- ThreadLocal使用后務(wù)必調(diào)用remove方法。
- 最簡單有效的方法是使用后將其移除。
關(guān)于InheritableThreadLocal
InheritableThreadLocal類是ThreadLocal類的子類。ThreadLocal中每個(gè)線程擁有它自己的值,與ThreadLocal不同的是,InheritableThreadLocal允許一個(gè)線程以及該線程創(chuàng)建的所有子線程都可以訪問它保存的值。
代碼示例
ThreadLocal使用
public class ThreadLocalTest { //第一種初始化方式 /** * 聲明為static是讓ThreadLocal實(shí)例隨著程序的結(jié)束才結(jié)束,這樣才不會(huì)讓GC回收了 * 聲明為final是讓ThreadLocal實(shí)例引用不會(huì)被替換,這樣子也不會(huì)因?yàn)楸惶鎿Q導(dǎo)致被GC回收 * 這兩個(gè)聲明都是為了避免作為key的ThreadLocal對(duì)象沒有外部強(qiáng)引用而導(dǎo)致被GC回收,從而導(dǎo)致內(nèi)存泄漏的問題,因?yàn)門hreadLocalMap<ThreadLocal, Object>中的ThreadLocal * 對(duì)象作為key是弱引用,會(huì)被GC回收。 */ private static final ThreadLocal<String> threadLocalStr = ThreadLocal.withInitial(() -> "fresh"); private static AtomicInteger intGen = new AtomicInteger(0); //第二種初始化方式 private static final ThreadLocal<Integer> threadLocalInt = new ThreadLocal<Integer>() { @Override public Integer initialValue() { return intGen.incrementAndGet(); } }; public static void main(String[] args) throws InterruptedException { ArrayList<Thread> threads = new ArrayList<>(); for (int i = 0; i < 2; i++) { Thread t = new Thread(() -> { try { System.out.println(Thread.currentThread().getName() + " " + threadLocalInt.get()); System.out.println(Thread.currentThread().getName() + " " + threadLocalStr.get()); TimeUnit.SECONDS.sleep(5); threadLocalStr.set("bojack horseman" + threadLocalInt.get()); System.out.println(Thread.currentThread().getName() + " " + threadLocalInt.get()); System.out.println(Thread.currentThread().getName() + " " + threadLocalStr.get()); } catch (InterruptedException e) { e.printStackTrace(); } finally { threadLocalInt.remove(); threadLocalStr.remove(); } }); t.start(); threads.add(t); } TimeUnit.SECONDS.sleep(2); System.out.println(threads); System.out.println(threadLocalStr); System.out.println(threadLocalInt); } /** * Thread-0 1 * Thread-1 2 * Thread-0 fresh * Thread-1 fresh * [Thread[Thread-0,5,main], Thread[Thread-1,5,main]] * java.lang.ThreadLocal$SuppliedThreadLocal@1ef7fe8e * cn.vv.schedule.test.ThreadLocalTest$1@6f79caec * Thread-1 2 * Thread-1 bojack horseman2 * Thread-0 1 * Thread-0 bojack horseman1 */ }
InheritableThreadLocal使用
public class InheritableThreadLocalTest { //第一種初始化方式 private static final InheritableThreadLocal<String> threadLocalStr = new InheritableThreadLocal<String>() { @Override public String initialValue() { return "fresh"; } }; private static AtomicInteger intGen = new AtomicInteger(0); //第二種初始化方式 private static final ThreadLocal<Integer> threadLocalInt = new ThreadLocal<Integer>() { @Override public Integer initialValue() { return intGen.incrementAndGet(); } }; public static void main(String[] args) throws InterruptedException { //如果是InheritableThreadLocal,則父線程創(chuàng)建的所有子線程都會(huì)復(fù)制一份父線程的線程變量,而不是去初始化一份線程變量 threadLocalStr.set("main"); ArrayList<Thread> threads = new ArrayList<>(); for (int i = 0; i < 2; i++) { Thread t = new Thread(() -> { try { System.out.println(Thread.currentThread().getName() + " " + threadLocalInt.get()); System.out.println(Thread.currentThread().getName() + " " + threadLocalStr.get()); TimeUnit.SECONDS.sleep(5); //子線程可以自由地改變自己的本地變量 threadLocalStr.set("bojack horseman" + threadLocalInt.get()); System.out.println(Thread.currentThread().getName() + " " + threadLocalInt.get()); System.out.println(Thread.currentThread().getName() + " " + threadLocalStr.get()); } catch (InterruptedException e) { e.printStackTrace(); } finally { threadLocalInt.remove(); threadLocalStr.remove(); } }); t.start(); threads.add(t); } TimeUnit.SECONDS.sleep(2); System.out.println(Thread.currentThread().getName() + " " + threadLocalStr.get()); } /** * Thread-0 2 * Thread-1 1 * Thread-0 main * Thread-1 main * main main * Thread-0 2 * Thread-0 bojack horseman2 * Thread-1 1 * Thread-1 bojack horseman1 */ }
參考
相關(guān)文章
解讀Jvm的內(nèi)存結(jié)構(gòu)與GC及jvm參數(shù)調(diào)優(yōu)
這篇文章主要介紹了解讀Jvm的內(nèi)存結(jié)構(gòu)與GC及jvm參數(shù)調(diào)優(yōu)方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-05-05java實(shí)現(xiàn)網(wǎng)頁爬蟲的示例講解
下面小編就為大家?guī)硪黄猨ava實(shí)現(xiàn)網(wǎng)頁爬蟲的示例講解。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-08-08Java 異常java.lang.NoSuchFieldException解決方案
這篇文章主要介紹了Java 異常java.lang.NoSuchFieldException解決方案,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-10-10springboot引用kettle實(shí)現(xiàn)對(duì)接oracle數(shù)據(jù)的示例代碼
這篇文章主要介紹了springboot引用kettle實(shí)現(xiàn)對(duì)接oracle數(shù)據(jù),其實(shí)kettle集成到springboot里面沒有多少代碼,這個(gè)功能最主要的還是ktr文件的編寫,只要ktr編寫好了,放到指定文件夾下,寫個(gè)定時(shí)任務(wù)就完事了,需要的朋友可以參考下2022-12-12Spring?Boot和Vue前后端分離項(xiàng)目架構(gòu)的全過程
前后端分離是目前互聯(lián)網(wǎng)開發(fā)中比較廣泛使用的開發(fā)模式,主要是將前端和后端的項(xiàng)目業(yè)務(wù)進(jìn)行分離,下面這篇文章主要給大家介紹了關(guān)于Spring?Boot和Vue前后端分離項(xiàng)目架構(gòu)的相關(guān)資料,需要的朋友可以參考下2022-04-04