日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

Java中ThreadLocal的一些理解

瀏覽:133日期:2022-08-19 17:38:29

前言

面試的時候被問到ThreadLocal的相關知識,沒有回答好(奶奶的,現在感覺問啥都能被問倒),所以我決定先解決這幾次面試中都遇到的高頻問題,把這幾個硬骨頭都能理解的透徹的說出來了,感覺最起碼不能總是一輪游。

ThreadLocal介紹

ThreadLocal是JDK1.2開始就提供的一個用來存儲線程本地變量的類。ThreadLocal中的變量是在每個線程中獨立存在的,當多個線程訪問ThreadLocal中的變量的時候,其實都是訪問的自己當前線程的內存中的變量,從而保證的變量的線程安全。

我們一般在使用ThreadLocal的時候都是為了解決線程中存在的變量競爭問題。其實解決這類問題,通常大家也會想到使用synchronized來加鎖解決。

例如在解決SimpleDateFormat的線程安全的時候。SimpleDateFormat是非線程安全的,它里面無論的是format()方法還是parse()方法,都有使用它自己內部的一個Calendar類的對象,format方法是設置時間,parse()方法里面是先調用Calendar的clear()方法,然后又調用了Calendar的set()方法(賦值),如果一個線程剛調用了set()進行賦值,這個時候又來了一個線程直接調用了clear()方法,那么這個parse()方法執行的結果就會有問題的。

解決辦法一將使用SimpleDateformat的方法加上synchronized,這樣雖然保證了線程安全,但卻降低了效率,同一時間只有一個線程能使用格式化時間的方法。

private static SimpleDateFormat simpleDateFormat = new SimpleDateFormat('yyyy-MM-dd HH:mm:ss');public static synchronized String formatDate(Date date){ return simpleDateFormat.format(date);}

解決辦法二將SimpleDateFormat的對象,放到ThreadLocal里面,這樣每個線程中都有一個自己的格式對象的副本了。互不干擾,從而保證了線程安全。

private static final ThreadLocal<SimpleDateFormat> simpleDateFormatThreadLocal = ThreadLocal.withInitial(() -> new SimpleDateFormat('yyyy-MM-dd HH:mm:ss'));public static String formatDate(Date date){ return simpleDateFormatThreadLocal.get().format(date);}

ThreadLocal的原理

我們先看一下ThreadLocal是怎么使用的。

ThreadLocal<Integer> threadLocal99 = new ThreadLocal<Integer>();threadLocal99.set(3);int num = threadLocal99.get();System.out.println('數字:'+num);threadLocal99.remove();System.out.println('數字Empty:'+threadLocal99.get());

運行結果:

數字:3數字Empty:null

使用起來很簡單,主要是將變量放到ThreadLocal里面,在線程執行過程中就可以取到,當執行完成后在remove掉就可以了,只要沒有調用remove()當前線程在執行過程中都是可以拿到變量數據的。因為是放到了當前執行的線程中,所以ThreadLocal中的變量值只能當前線程來使用,從而保證的了線程安全(當前線程的子線程其實也是可以獲取到的)。

來看一下ThreadLocal的set()方法源碼

public void set(T value) { // 獲取當前線程 Thread t = Thread.currentThread(); // 獲取ThreadLocalMap ThreadLocal.ThreadLocalMap map = getMap(t); // ThreadLocalMap 對象是否為空,不為空則直接將數據放入到ThreadLocalMap中 if (map != null) map.set(this, value); else createMap(t, value); // ThreadLocalMap對象為空,則先創建對象,再賦值。}

我們看到變量都是存放在了ThreadLocalMap這個變量中的。那么ThreadLocalMap又是怎么來的呢?

ThreadLocalMap getMap(Thread t) { return t.threadLocals;}

public class Thread implements Runnable {... .../* ThreadLocal values pertaining to this thread. This map is maintained * by the ThreadLocal class. */ ThreadLocal.ThreadLocalMap threadLocals = null; ... ...}

通過上面的源碼,我們發現ThreadLocalMap變量是當前執行線程中的一個變量,所以說,ThreadLocal中存放的數據其實都是放到了當前執行線程中的一個變量里面了。也就是存儲在了當前的線程對象里了,別的線程里面是另一個線程對象了,拿不到其他線程對象中的數據,所以數據自然就隔離開了。

那么ThreadLocalMap是怎么存儲數據的呢?ThreadLocalMap 是ThreadLocal類里的一個內部類,雖然類的名字上帶著Map但卻沒有實現Map接口,只是結構和Map類似而已。

Java中ThreadLocal的一些理解

ThreadLocalMap內部其實是一個Entry數組,Entry是ThreadLocalMap中的一個內部類,繼承自WeakReference,并將ThreadLocal類型的對象設置為了Entry的Key,以及對Key設置成弱引用。ThreadLocalMap的內部數據結構,就大概是這樣的key,value組成的Entry的數組集合。

Java中ThreadLocal的一些理解

和真正的Map還是有區別的,沒有鏈表了,這樣在解決key的hash沖突的時候措施肯定就和HashMap不一樣了。一個線程中是可以創建多個ThreadLocal對象的,多個ThreadLocal對象就會存放多個數據,那么在ThreadLocalMap中就會以數組的形式存放這些數據。我們來看一下具體的ThreadLocalMap的set()方法的源碼

/** * Set the value associated with key. * @param key the thread local object * @param value the value to be set */private void set(ThreadLocal<?> key, Object value) { // We don’t use a fast path as with get() because it is at // least as common to use set() to create new entries as // it is to replace existing ones, in which case, a fast // path would fail more often than not. Entry[] tab = table; int len = tab.length; // 定位在數組中的位置 int i = key.threadLocalHashCode & (len-1); for (Entry e = tab[i]; e != null; e = tab[i = nextIndex(i, len)]) { ThreadLocal<?> k = e.get(); // 如果當前位置不為空,并且當前位置的key和傳過來的key相等,那么就會覆蓋當前位置的數據 if (k == key) { e.value = value; return; } // 如果當前位置為空,則初始化一個Entry對象,放到當前位置。 if (k == null) { replaceStaleEntry(key, value, i); return; } } // 如果當前位置不為空,并且當前位置的key也不等于要賦值的key ,那么將去找下一個空位置,直接將數據放到下一個空位置處。 tab[i] = new Entry(key, value); int sz = ++size; if (!cleanSomeSlots(i, sz) && sz >= threshold) rehash();}

我們從set()方法中可以看到,處理邏輯有四步。

第一步先根據Threadlocal對象的hashcode和數組長度做與運算獲取數據應該放在當前數組中的位置。 第二步就是判斷當前位置是否為空,為空的話就直接初始化一個Entry對象,放到當前位置。 第三步如果當前位置不為空,而當前位置的Entry中的key和傳過來的key一樣,那么直接覆蓋掉當前位置的數據。 第四步如果當前位置不為空,并且當前位置的Entry中的key和傳過來的key 也不一樣,那么就會去找下一個空位置,然后將數據存放到空位置(數組超過長度后,會執行擴容的);

在get的時候也是類似的邏輯,先通過傳入的ThreadLocal的hashcode獲取在Entry數組中的位置,然后拿當前位置的Entry的Key和傳入的ThreadLocal對比,相等的話,直接把數據返回,如果不相等就去判斷和數組中的下一個值的key是否相等。。。

private Entry getEntry(ThreadLocal<?> key) { int i = key.threadLocalHashCode & (table.length - 1); Entry e = table[i]; if (e != null && e.get() == key) return e; else return getEntryAfterMiss(key, i, e);}

/** * Version of getEntry method for use when key is not found in * its direct hash slot. * * @param key the thread local object * @param i the table index for key’s hash code * @param e the entry at table[i] * @return the entry associated with key, or null if no such */private Entry getEntryAfterMiss(ThreadLocal<?> key, int i, Entry e) { Entry[] tab = table; int len = tab.length; while (e != null) { ThreadLocal<?> k = e.get(); if (k == key) return e; if (k == null) expungeStaleEntry(i); else i = nextIndex(i, len); e = tab[i]; } return null;}

我們上文一直說,ThreadLocal是保存在單個線程中的數據,每個線程都有自己的數據,但是實際ThreadLocal里面的真正的對象數據,其實是保存在堆里面的,而線程里面只是存儲了對象的引用而已。并且我們在使用的時候通常需要在上一個線程執行的方法的上下文共享ThreadLocal中的變量。例如我的主線程是在某個方法中執行代碼呢,但是這個方法中有一段代碼時新創建了一個線程,在這個線程里面還使用了我這個正在執行的方法里面的定義的ThreadLocal里面的變量。這個時候,就是需要從新線程里面調用外面線程的數據,這個就需要線程間共享了。這種子父線程共享數據的情況,ThreadLocal也是支持的。例如:

ThreadLocal threadLocalMain = new InheritableThreadLocal(); threadLocalMain.set('主線程變量'); Thread t = new Thread() { @Override public void run() { super.run(); System.out.println( '現在獲取的變量是 =' + threadLocalMain.get()); } }; t.start();

運行結果:

現在獲取的變量是 =主線程變量

上面這樣的代碼就能實現子父線程共享數據的情況,重點是使用InheritableThreadLocal來實現的共享。那么它是怎么實現數據共享的呢?在Thread類的init()方法中有這么一段代碼:

if (inheritThreadLocals && parent.inheritableThreadLocals != null) this.inheritableThreadLocals =ThreadLocal.createInheritedMap(parent.inheritableThreadLocals);

這段代碼的意思是,在創建線程的時候,如果當前線程的inheritThreadLocals變量和父線程的inheritThreadLocals變量都不為空的時候,會將父線程的inheritThreadLocals變量中的數據,賦給當前線程中的inheritThreadLocals變量。

ThreadLocal的內存泄漏問題上文我們也提到過,ThreadLocal中的ThreadLocalMap里面的Entry對象是繼承自WeakReference類的,說明Entry的key是一個弱引用。

Java中ThreadLocal的一些理解

弱引用是用來描述那些非必須的對象,弱引用的對象,只能生存到下一次垃圾收集發生為止。當垃圾收集器開始工作,無論當前內存是否足夠,都會回收掉只被弱引用關聯的對象。

這個弱引用還是ThreadLocal對象本身,所以一般在線程執行完成后,ThreadLocal對象就會變成null了,而為null的弱引用對象,在下一次GC的時候就會被清除掉,這樣Entry的Key的內存空間就被釋放出來了,但是Entry的value還在占用的內存,如果線程是被復用的(例如線程池中的線程),那么這里面的value值就會越來越多,最終就導致了內存泄漏。

防止內存泄漏的辦法就是在每次使用完ThreadLocal的時候都去執行以下remove()方法,就可以把key和value的空間都釋放了。

那既然容易產生內存泄漏,為什么還要設置成弱引用的呢?如果正常情況下應該是強引用,但是強引用只要引用關系還在就一直不會被回收,所以如果線程被復用了,那么Entry中的Key和Value都不會被回收,這樣就造成了Key和Value都會發生內存泄漏了。

以上就是Java中ThreadLocal的一些理解的詳細內容,更多關于Java ThreadLocal的資料請關注好吧啦網其它相關文章!

標簽: Java
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
欧美黄色网页| 亚洲永久字幕| 日韩国产在线观看| 只有精品亚洲| 国产视频一区三区| 亚洲欧美不卡| 在线精品福利| 日本精品一区二区三区在线观看视频 | 欧美一区二区三区高清视频| 国产精品二区不卡| 精品免费视频| 高清不卡亚洲| 欧美三级精品| 久久蜜桃精品| 中文亚洲免费| 亚洲一区二区三区久久久| 欧美日韩黑人| 亚洲欧美日韩在线观看a三区| 麻豆精品网站| 91亚洲无吗| 欧美黄色一区二区| 久久久久久一区二区| 日韩大片在线| 精品一区在线| 亚洲另类视频| 日本中文字幕不卡| 久久精品国产久精国产| 中文字幕在线免费观看视频| 日韩精品午夜| 蜜桃视频免费观看一区| 欧美性www| 国产精品igao视频网网址不卡日韩 | 久久久噜噜噜| 国产视频一区三区| 日本成人在线一区| 麻豆成人综合网| 97精品在线| 中文亚洲免费| 欧美日本二区| 999久久久国产精品| 国产精品日韩| 亚洲免费专区| 另类欧美日韩国产在线| 婷婷成人在线| 日本99精品| 麻豆视频在线观看免费网站黄| 国产99久久久国产精品成人免费| 亚洲九九精品| 久久99高清| 好吊一区二区三区| 国产精品伊人| 久久中文字幕二区| 国产日产一区| 欧美日韩亚洲在线观看| 欧美久久精品| 今天的高清视频免费播放成人| 亚洲一区导航| 国产成年精品| 男女男精品视频网| 国产一区二区三区四区二区| 99国产精品久久久久久久| 国产毛片久久久| 欧美91视频| 国产激情一区| 丝袜美腿一区二区三区| 精品国产亚洲一区二区三区| 在线综合欧美| 日本а中文在线天堂| 亚洲精品免费观看| 日韩成人精品一区二区| 婷婷综合成人| 国产99在线| 日韩一区精品| 999久久久精品国产| 欧美经典一区| 喷白浆一区二区| 久久久久久一区二区| 亚久久调教视频| 91tv亚洲精品香蕉国产一区| 91嫩草精品| 图片区亚洲欧美小说区| 久久国产视频网| 乱人伦精品视频在线观看| 伊人久久在线| 里番精品3d一二三区| 在线精品视频一区| 欧美jjzz| 精品视频网站| 亚洲精品系列| 国产女优一区| 亚洲国产专区校园欧美| 国产成人免费精品| 国产人成精品一区二区三| 国产精品婷婷| 久久精品国产www456c0m| 麻豆视频观看网址久久| 日韩欧美久久| 三级欧美在线一区| 亚洲小说欧美另类婷婷| 日本一区二区免费高清| 国产精品一国产精品k频道56| 天堂成人免费av电影一区| 97精品一区二区| 精品亚洲a∨一区二区三区18| 亚洲另类av| 日韩视频中文| 精品一区三区| 久久亚洲成人| 久久精品1区| 日产精品一区二区| 精品视频免费| 久久久精品国产**网站| 国产精品久久乐| 国产日韩欧美三级| 热久久久久久| 久久国产麻豆精品| 日韩二区三区四区| 免费在线观看成人| 丝瓜av网站精品一区二区| 婷婷六月综合| 香蕉国产精品| 亚洲特级毛片| 国内激情久久| 不卡中文字幕| 在线看片不卡| 亚洲欧美激情诱惑| 香蕉成人久久| 一区二区国产精品| 中文无码久久精品| 日韩精品a在线观看91| 蜜桃视频在线观看一区二区| 久久午夜影视| 美女被久久久| 一区二区三区国产在线| 日韩一区二区三免费高清在线观看 | 日本精品久久| 日韩精彩视频在线观看| 日韩精品久久理论片| 日本电影久久久| 国产精品亚洲欧美一级在线| 国产精品2区| 国产一区二区三区四区大秀| 福利片在线一区二区| 福利片在线一区二区| 欧美日韩免费观看视频| 免费av一区| 日韩在线卡一卡二| 奇米色欧美一区二区三区| 国产精品xxx| 日本一区二区高清不卡| 久久精品国产68国产精品亚洲| 国产一区日韩欧美| 久久大逼视频| 欧美日韩亚洲国产精品| 精品国产成人| 欧美色图国产精品| 亚洲一区欧美激情| 日韩成人精品一区二区三区 | 日韩影院免费视频| 日本成人在线视频网站| 免费在线日韩av| 色一区二区三区| 日韩午夜av在线| 亚洲精品人人| 久久精品国产亚洲一区二区三区| 中文字幕人成乱码在线观看| 亚洲精品123区| 日本成人在线视频网站| 风间由美中文字幕在线看视频国产欧美| 国产va在线视频| 免费视频国产一区| 日本综合精品一区| 精品精品久久| 91久久午夜| 欧美1区二区| 激情综合网站| 国产调教一区二区三区| 黄色网一区二区| 今天的高清视频免费播放成人| 亚洲人成精品久久久| 亚洲激情中文| 国产欧美大片| 91精品国产调教在线观看| 中文字幕av一区二区三区四区| 青草av.久久免费一区| 在线手机中文字幕| 亚洲视频电影在线| 久久精品免视看国产成人| 好吊视频一区二区三区四区| 国产亚洲欧美日韩精品一区二区三区 | 91精品国产调教在线观看 | 国产欧美一区二区三区国产幕精品| 精品72久久久久中文字幕| 欧美+日本+国产+在线a∨观看| 日韩一区二区三区四区五区| 天堂√8在线中文| 亚洲精品在线国产| 欧美一级鲁丝片| 日本伊人久久| 亚洲高清毛片|