AVt天堂网 手机版,亚洲va久久久噜噜噜久久4399,天天综合亚洲色在线精品,亚洲一级Av无码毛片久久精品

當前位置:首頁 > 科技  > 軟件

?Thread Local深度解析,你學會了嗎?

來源: 責編: 時間:2024-02-05 17:18:42 225觀看
導讀今天,有個朋友問我說他想在并發條件下統計接口的耗時以及日期,并做一個記錄在最后統一保存,這里我就直接想到了ThreadLocal,其實我用ThreadLocal的場景還挺多的,畢竟項目需要,其實一直都想對ThreadLocal做一個總結,擇日不如

今天,有個朋友問我說他想在并發條件下統計接口的耗時以及日期,并做一個記錄在最后統一保存,這里我就直接想到了ThreadLocal,其實我用ThreadLocal的場景還挺多的,畢竟項目需要,其實一直都想對ThreadLocal做一個總結,擇日不如撞日就現在動手吧。40i28資訊網——每日最新資訊28at.com

ThreadLocal概念

ThreadLocal也叫做本地線程變量,ThreadLocal中填充的是當前線程的變量,該變量對其他線程是隔離的,ThreadLocal在每個線程中都創建了一個變量副本,所以每個線程中的ThreadLocal都是一個獨立的副本,自己可以訪問自己線程內部的副本變量互不干擾。40i28資訊網——每日最新資訊28at.com

ThreadLocal使用場景

ThreadLocal的使用也要看情況來定,按個人理解ThreadLocal大致會使用到以下場景:40i28資訊網——每日最新資訊28at.com

  • 需要全局獲取變量(保證這個變量在全局中的一致性)
  • 需要解決線程安全的場景(例如:記錄每個請求的一些信息,保存到日志表中)
  • 父子線程需要共享數據(例如:需要子線程的結果回調給父線程,如何保存它的唯一性)

說白了ThreadLocal就是做數據隔離,每條線程的ThreadLocal都是隔離的互不干擾,其實就是為了防止多線程環境下變量被其他線程篡改,只要記住這點在工作中什么場景下會使用到就一目了然了。40i28資訊網——每日最新資訊28at.com

實際上Spring就是采用了Threadlocal來實現單個線程中的數據庫操作使用的是同一個數據庫連接,采用Threadlocal可以使業務層使用事務的時候不需要去管理connection對象,通過傳播級別就能管理多個事務配置之間的切換,掛起和恢復。40i28資訊網——每日最新資訊28at.com

Spring框架里面就是用的ThreadLocal來實現這種隔離,主要是在TransactionSynchronizationManager這個類里面,代碼如下所示:40i28資訊網——每日最新資訊28at.com

private static final Log logger = LogFactory.getLog(TransactionSynchronizationManager.class);private static final ThreadLocal<Map<Object, Object>> resources =  new NamedThreadLocal<>("Transactional resources");private static final ThreadLocal<Set<TransactionSynchronization>> synchronizations =  new NamedThreadLocal<>("Transaction synchronizations");private static final ThreadLocal<String> currentTransactionName =  new NamedThreadLocal<>("Current transaction name");

40i28資訊網——每日最新資訊28at.com

注意:在Spring5.2以后的版本Spring事務隔離從ThreadLocal換成了Mono響應式編程來實現隔離。40i28資訊網——每日最新資訊28at.com

圖片圖片40i28資訊網——每日最新資訊28at.com

ThreadLocal源碼分析

圖片圖片40i28資訊網——每日最新資訊28at.com

從源碼上看其實ThreadLocal的set方法并不復雜
  1. 獲取當前線程對象Thread.currentThread();
  2. 獲取線程變量ThreadLocalMap map = getMap(t);
  3. 如果不為空則賦值map.set(this,value);
  4. 如果為空,初始化該線程對象的map變量,其中key為當前的threadlocal變量createMap(t,value);
再看看ThreadLocal的get方法

圖片圖片40i28資訊網——每日最新資訊28at.com

圖片圖片40i28資訊網——每日最新資訊28at.com

  • 返回當前線程變量的副本中的值,如果該變量沒有當前線程的值,則先調用initialValue方法的返回值
  • initialValue方法中繼續獲取當前線程變量(Key為當前線程)而Value設置為null
  • 如果當前線程副本變量為空那么重新創建當前線程的Map(Key為當前線程,Value為null)
ThreadLocal如何做到線程隔離?

上面分析了ThreadLocal的set()和get()源碼,在通過get()方法獲取當前線程中副本變量為null那么直接創建一個ThreadLocalMap:40i28資訊網——每日最新資訊28at.com

圖片圖片40i28資訊網——每日最新資訊28at.com

從這里入手,看一下t.threadLocals40i28資訊網——每日最新資訊28at.com

圖片圖片40i28資訊網——每日最新資訊28at.com

注釋說得很清楚:ThreadLocal屬于當前這個線程的。40i28資訊網——每日最新資訊28at.com

注意:這個ThreadLocalMap是一個靜態內部類40i28資訊網——每日最新資訊28at.com

圖片圖片40i28資訊網——每日最新資訊28at.com

ThreadLocalMap is a customized hash map suitable only for maintaining thread local values. No operations are exported outside of the ThreadLocal class. The class is package private to allow declaration of fields in class Thread. To help deal with very large and long-lived usages, the hash table entries use WeakReferences for keys. However, since reference queues are not used, stale entries are guaranteed to be removed only when the table starts running out of space.40i28資訊網——每日最新資訊28at.com

到此為止其實ThreadLocal的數據隔離的真相就出來了,說白了每個線程Thread都維護了自己的一個threadLocals變量,當線程創建ThreadLocal的時候,實際上數據是存在自己的線程Thread的threadLocals變量里面,可以看出來這個ThreadLocalMap這個類只有一份,在線程中,所以實現了線程之間的隔離。40i28資訊網——每日最新資訊28at.com

ThreadLocalMap底層原理

圖片圖片40i28資訊網——每日最新資訊28at.com

雖然看著ThreadLocalMap很像是HashMap,實際上并沒有實現Map接口,而是它的內部類Entry繼承了WeakReference這個弱引用,也就是說不存在鏈表的關系了。40i28資訊網——每日最新資訊28at.com

接下來我們來看一下ThreadLocalMap的set()方法(這里圖片沒有截全):40i28資訊網——每日最新資訊28at.com

圖片圖片40i28資訊網——每日最新資訊28at.com

ThreadLocalMap在存儲的時候每次都會給每一個ThreadLocal對象一個threadLocalHashCode,在插入過程中,根據ThreadLocal對象的hash值,定位到table中的位置i,int i = key.threadLocalHashCode & (len - 1);40i28資訊網——每日最新資訊28at.com

接下來判斷如果當前位置為null,就初始化一個Entry對象放在位置上40i28資訊網——每日最新資訊28at.com

圖片圖片40i28資訊網——每日最新資訊28at.com

如果當前位置i不為空,又剛好這個Entry對象的key正好是即將設置的key,那么就覆蓋Entry中的value40i28資訊網——每日最新資訊28at.com

圖片圖片40i28資訊網——每日最新資訊28at.com

如果位置i不為null并且key不等于 entry,那么就找下一個空位置,直到位置為空為止然后存放。40i28資訊網——每日最新資訊28at.com

在get的時候就會根據ThreadLocal對象的Hash值,定位到相應位置,然后判斷該位置Entry對象中的key是否和get的key一致,如果不一致,就判斷下個位置。40i28資訊網——每日最新資訊28at.com

如何共享ThreadLocal中的數據?

使用 InheritableThreadLocal可以實現多個線程訪問ThreadLocal的值。40i28資訊網——每日最新資訊28at.com

問題是它們之間是如何實現傳遞的?

其實邏輯很簡單,繼續看Thread的源碼,看下初始化的時候Thread.init做了什么操作:40i28資訊網——每日最新資訊28at.com

圖片圖片40i28資訊網——每日最新資訊28at.com

如果線程的inheritThreadLocals變量不為空的話,并且父線程的inheritThreadLocals不為空的話,就把線程的inheritThreadLocals給當前線程的inheritThreadLocals40i28資訊網——每日最新資訊28at.com

圖片圖片40i28資訊網——每日最新資訊28at.com

關于ThreadLocal內存泄露

ThreadLocal使用不當也會出現問題:那就是內存泄露40i28資訊網——每日最新資訊28at.com

繼續查看最開始存儲數據的Entry類的源碼40i28資訊網——每日最新資訊28at.com

圖片圖片40i28資訊網——每日最新資訊28at.com

其實文檔已經說得很直白了:40i28資訊網——每日最新資訊28at.com

Note that null keys (i.e. entry.get()* == null 如果 key threadlocal 為 null 了,這個 entry 就可以清除了。40i28資訊網——每日最新資訊28at.com

ThreadLocal是一個弱引用,當為null時,會被當成垃圾回收 。40i28資訊網——每日最新資訊28at.com

造成內存泄露的原因在于ThreadLocal為null,也就是要被垃圾回收器回收了,但是此時我們的ThreadLocalMap(thread 的內部屬性)生命周期和Thread的一樣,它不會回收,這時候就出現了一個現象。那就是ThreadLocalMap的key沒了,但是value還在,這就造成了內存泄漏。40i28資訊網——每日最新資訊28at.com

再詳細點來說,ThreadLocal在沒有外部強引用時,發生GC時會被回收,如果創建ThreadLocal的線程一直持續運行,那么這個Entry對象中的value就有可能一直得不到回收,發生內存泄露。40i28資訊網——每日最新資訊28at.com

就比如線程池里面的線程,線程都是復用的,那么之前的線程實例處理完之后,出于復用的目的線程依然存活,所以,ThreadLocal設定的value值被持有,導致內存泄露。40i28資訊網——每日最新資訊28at.com

按照道理一個線程使用完,ThreadLocalMap是應該要被清空的,但是現在線程被復用了。40i28資訊網——每日最新資訊28at.com

解決辦法:

每次在使用完ThreadLocal的時候一定要remove。40i28資訊網——每日最新資訊28at.com

為什么ThreadLocal要使用弱引用?

如果使用強引用,當ThreadLocal 對象的引用(強引用)被回收了,ThreadLocalMap本身依然還持有ThreadLocal的強引用,如果沒有手動刪除這個key ,則ThreadLocal不會被回收,所以只要當前線程不消亡,ThreadLocalMap引用的那些對象就不會被回收, 可以認為這導致Entry內存泄漏。40i28資訊網——每日最新資訊28at.com

  • 強引用:普通的引用,強引用指向的對象不會被回收。
  • 軟引用:僅有軟引用指向的對象,只有發生gc且內存不足,才會被回收。
  • 弱引用:僅有弱引用指向的對象,只要發生gc就會被回收。

本文鏈接:http://www.tebozhan.com/showinfo-26-74189-0.html?Thread Local深度解析,你學會了嗎?

聲明:本網頁內容旨在傳播知識,若有侵權等問題請及時與本網聯系,我們將在第一時間刪除處理。郵件:2376512515@qq.com

上一篇: 巴可可立享重磅新品 —— 無線視頻會議一體機,多合一,“會”更強

下一篇: Go 中如何檢查文件是否存在?可能產生競態條件?

標簽:
  • 熱門焦點
  • MIX Fold3包裝盒泄露 新機本月登場

    小米的全新折疊屏旗艦MIX Fold3將于本月發布,近日該機的真機包裝盒在網上泄露。從圖上來看,新的MIX Fold3包裝盒在外觀設計方面延續了之前的方案,變化不大,這也是目前小米旗艦
  • 天貓精靈Sound Pro體驗:智能音箱沒有音質?來聽聽我的

    這幾年除了手機作為智能生活終端最主要的核心之外,第二個可以成為中心點的產品是什么?——是智能音箱。 手機在執行命令的時候有兩種操作方式,手和智能語音助手,而智能音箱只
  • 6月安卓手機性價比榜:Note 12 Turbo斷層式碾壓

    6月份有一個618,雖然這是京東周年慶的日子,但別的電商也都不約而同的跟進了,反正促銷沒壞處,廠商和用戶都能滿意。618期間一些產品也出現了歷史低價,那么各個價位段的產品性價比
  • 6月安卓手機性能榜:vivo/iQOO霸占旗艦排行榜前三

    2023年上半年已經正式過去了,我們也迎來了安兔兔V10版本,在新的驍龍8Gen3和天璣9300發布之前,性能榜的榜單大體會以驍龍8Gen2和天璣9200+為主,至于那顆3.36GHz的驍龍8Gen2領先
  • 5月iOS設備好評榜:iPhone 14僅排第43?

    來到新的一月,安兔兔的各個榜單又重新匯總了數據,像安卓陣營的榜單都有著比較大的變動,不過iOS由于設備的更新換代并沒有那么快,所以相對來說變化并不大,特別是iOS好評榜,老款設
  • 自動化在DevOps中的力量:簡化軟件開發和交付

    自動化在DevOps中扮演著重要角色,它提升了DevOps的效能。通過自動化工具和方法,DevOps團隊可以實現以下目標:消除手動和重復性任務。簡化流程。在整個軟件開發生命周期中實現更
  • 猿輔導與新東方的兩種“歸途”

    作者|卓心月 出品|零態LT(ID:LingTai_LT)如何成為一家偉大企業?答案一定是對&ldquo;勢&rdquo;的把握,這其中最關鍵的當屬對企業戰略的制定,且能夠站在未來看現在,即使這其中的
  • iQOO Neo8系列今日官宣:首發天璣9200+ 全球安卓最強芯!

    在昨日舉行的的聯發科新一代旗艦芯片天璣9200+的發布會上,iQOO官方也正式宣布,全新的iQOO Neo8系列新品將全球首發搭載這款當前性能最強大的移動平臺
  • Android 14發布:首批適配機型公布

    5月11日消息,谷歌在今天凌晨舉行了I/O大會,本次發布會谷歌帶來了自家的AI語言模型PaLM 2、谷歌Pixel Fold折疊屏、谷歌Pixel 7a手機,同時發布了Androi
Top