摘要:而應用場景更多是想共享一個變量,但是該變量又不是線程安全的,那么可以用維護一個線程一個實例。因為創建這個對象本身很費時的,而且我們也知道本身不是線程安全的,也不能緩存一個共享的實例,為此我們想到使用來給每個線程緩存一個實例,提高性能。
很多人都知道java中有ThreadLocal這個類,但是知道ThreadLocal這個類具體有什么作用,然后適用什么樣的業務場景還是很少的。今天我就嘗試以自己的理解,來講解下ThreadLocal類的內部實現和應用場景,如果有什么不對之處,還望大家指正。
首先明確一個概念,那就是ThreadLocal并不是用來并發控制訪問一個共同對象,而是為了給每個線程分配一個只屬于該線程的對象(這么粗暴的解釋可能還不太準確),更準確的說是為了實現線程間的數據隔離。而ThreadLocal應用場景更多是想共享一個變量,但是該變量又不是線程安全的,那么可以用ThreadLocal維護一個線程一個實例。有時候ThreadLocal也可以用來避免一些參數傳遞,通過ThreadLocal來訪問對象。
首先我們先看下ThreadLocal(jdk1.7)內部的實現:
ThreadLocal get方法
public T get() { Thread t = Thread.currentThread(); ThreadLocalMap map = getMap(t); if (map != null) { ThreadLocalMap.Entry e = map.getEntry(this); if (e != null) return (T)e.value; } return setInitialValue(); }
ThreadLocal set方法
public void set(T value) { Thread t = Thread.currentThread(); ThreadLocalMap map = getMap(t); if (map != null) map.set(this, value); else createMap(t, value); }
get和set方法是ThreadLocal類中最常用的兩個方法。
get方法
代碼很容易理解,首先我們通過Thread.currentThread得到當前線程,然后獲取當前線程的threadLocals變量,這個變量就是ThreadLocalMap類型的。然后根據當前的ThreadLocal實例作為key,獲取到Entry對象。
set方法
代碼同樣很容易理解。同樣根據Thread.currentThread得到當前線程,如果當前線程存在threadLocals這個變量不為空,那么根據當前的ThreadLocal實例作為key尋找在map中位置,然后用新的value值來替換舊值。
在ThreadLocal這個類中比較引人注目的應該是ThreadLocal->ThreadLocalMap->Entry這個類。這個類繼承自WeakReference。關于弱引用的知識,以后我會抽時間寫篇文章來介紹下。
最近在我們的web項目中servlet需要頻繁創建SimpleDateFormat這個對象,進行日期格式化。因為創建這個對象本身很費時的,而且我們也知道SimpleDateFormat本身不是線程安全的,也不能緩存一個共享的SimpleDateFormat實例,為此我們想到使用ThreadLocal來給每個線程緩存一個SimpleDateFormat實例,提高性能。同時因為每個Servlet會用到不同pattern的時間格式化類,所以我們對應每一種pattern生成了一個ThreadLocal實例。
DateFormatFactory
public class DateFormatFactory { private static final Map> pattern2ThreadLocal; static { DatePattern[] patterns = DatePattern.values(); int len = patterns.length; pattern2ThreadLocal = new HashMap >(len); for (int i = 0; i < len; i++) { DatePattern datePattern = patterns[i]; final String pattern = datePattern.pattern; pattern2ThreadLocal.put(datePattern, new ThreadLocal () { @Override protected DateFormat initialValue() { return new SimpleDateFormat(pattern); } }); } } //獲取DateFormat public static DateFormat getDateFormat(DatePattern pattern) { ThreadLocal threadDateFormat = pattern2ThreadLocal.get(pattern); //不需要判斷threadDateFormat是否為空 return threadDateFormat.get(); }
}
DatePattern 枚舉類
public enum DatePattern { TimePattern("yyyy-MM-dd HH:mm:ss"), DatePattern("yyyy-MM-dd"), public String pattern; private DatePattern(String pattern) { this.pattern = pattern; }
}
這樣我們就可以每次調用DateFormatFactory.getDateFormat獲取到對應的時間格式化類了。之前我們提到使用ThreadLocal同時可以避免參數傳遞。假如我們這個Servlet要調用到其他類的方法,而且方法內需要使用時間格式化類。按照正常情況下我們把該時間格式化類作為參數進行傳遞,但如果有了ThreadLocal這個類,我們可以不需要作為參數傳遞了,可以在方法類通過ThreadLocal得到時間格式化類。當然代碼的通用性就差了。
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://specialneedsforspecialkids.com/yun/64081.html
摘要:多線程類庫對于共享數據的讀寫控制主要采用鎖機制保證線程安全,本文所要探究的則采用了一種完全不同的策略。所以出現內存泄露的前提必須是持有的線程一直存活,這在使用線程池時是很正常的,在這種情況下一直不會被,因為 Java 多線程類庫對于共享數據的讀寫控制主要采用鎖機制保證線程安全,本文所要探究的 ThreadLocal 則采用了一種完全不同的策略。ThreadLocal 不是用來解決共享數...
摘要:什么是,簡單翻譯過來就是本地線程,但是直接這么翻譯很難理解的作用,如果換一種說法,可以稱為線程本地存儲。魔數的選取和斐波那契散列有關,對應的十進制為。而斐波那契散列的乘數可以用如果把這個值給轉為帶符號的,則會得到。 什么是ThreadLocal ThreadLocal,簡單翻譯過來就是本地線程,但是直接這么翻譯很難理解ThreadLocal的作用,如果換一種說法,可以稱為線程本地存儲。...
摘要:通過將保存在中,每個線程都會擁有屬于自己的,代碼如下所示然后你就可以安心地調用了,不用考慮線程安全問題。這樣設計的好處就是,當線程死掉之后,沒有強引用,方便收集器回收。 前言 想必大家都對Threadlocal很熟悉吧,今天我們就一起來深入學習一下。Threadlocal我更傾向于將其翻譯成線程局部變量。它有什么用處呢?Threadlocal對象通常用于防止對可變的單實例變量或全局變量...
摘要:另載于是個很爽的東西,線程安全,能當全局變量來用別。第一家公司,使用框架老技術,現代人可以理解為類似,對每個請求都套上,進入時把寫入,返回或拋注意時清理。第二家公司,某次引入一個設計,也用了來傳遞上下文信息,有的地方沒能清掉。 另載于 http://www.qingjingjie.com/blogs/12 ThreadLocal是個很爽的東西,線程安全,能當全局變量來用(別!)。 上一...
閱讀 955·2023-04-25 23:50
閱讀 1954·2021-11-19 09:40
閱讀 598·2019-08-30 13:50
閱讀 2727·2019-08-29 17:11
閱讀 1041·2019-08-29 16:37
閱讀 2986·2019-08-29 12:54
閱讀 2792·2019-08-28 18:17
閱讀 2636·2019-08-26 16:55