成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

ThreadLocal奪命4問

開發 前端
ThreadLocal作為實現“線程封閉”的最主要的編程手段,經常被使用。比如,比如,傳統的SimpleDateFormat,不是線程安全的。

[[376813]]

本文轉載自微信公眾號「小姐姐味道」,作者小姐姐養的狗 。轉載本文請聯系小姐姐味道公眾號。  

閱讀本文需要首先大體了解ThreadLocal。不啰嗦,直接進入正題。

標簽:【各種級別】【Java】【源碼】

1. 問

連環四問:

  • ThreadLocal的原理?
  • 內存泄漏的原因?
  • InheritableThreadLocal用過嗎?
  • Netty的FastThreadLocal是什么?

2. 分析

ThreadLocal作為實現“線程封閉”的最主要的編程手段,經常被使用。比如,比如,傳統的SimpleDateFormat,不是線程安全的。如果你聲明成全局變量,在并發環境下就會產生時間錯亂。一種好的解決方式,就是使用ThreadLocal。

ThreadLocal使用非常廣泛。比如,Spring的事務管理,就是通過它實現的。但它的弱點也是有的,不能透傳(不能被子線程獲取),所以催生了InheritableThreadLocal,甚至更高級的封裝庫。

3. 答

3.1 ThreadLocal的原理?

看過源碼就不難回答。如下圖(這張圖最易懂),ThreadLocal的get和remove方法,只不過是一個使用的快捷方式。它的真正數據,是存在于線程中的一個叫做ThreadLocalMap的結構里。

一個ThreadLocal的值,會根據線程的不同,分散在N個線程中。所以獲取ThreadLocal的Value,有兩個步驟。

  • 第一步,根據線程獲取Map
  • 第二部,根據自身從Map中獲取值,所以它的this就是Map的Key

這沒什么原理。這就是一個為了照顧編碼習慣的數據結構。

3.2 內存泄漏的原因?

嚴格來說,ThreadLocal沒有內存泄漏問題。有的話,那就是你忘記執行remove方法。這是不正確使用引起的。

這和其他一些內存泄漏的問題是一致的,比如:

  • 流沒有關閉
  • 連接沒有斷開
  • 濫用static map

為什么會有泄漏問題?

如果你不調用remove方法的話,ThreadLocal所對應的值,就會存在,一直到當前線程的銷毀。

眾所周知,線程的生命周期都比較長,加上現在普遍使用的線程池,會讓線程的生命更加長。不remove,當然不會釋放。這和Key,到底是不是弱引用,關系不大。

那這種情況,屬不屬于泄漏問題,是一個咬字眼的問題。面試的過程是探討,并不一定要標準的答案。

比起內存泄漏問題,線程池所引起的數據錯亂問題,更加應該引起關心。因為放在ThreadLocal的數據,肯定不會很大,泄漏頂多占用一點內存而已;而數據錯亂,可是會引起業務Bug的。

3.3 InheritableThreadLocal用過嗎?

InheritableThreadLocal在父子線程傳遞值的時候用到過,解決了threadlocal不能在父子線程間傳值的問題。

這個在本質上,還是通過Thread來實現的。通過兩個Map來進行屬性拷貝。

  1. /* ThreadLocal values pertaining to this thread. This map is maintained 
  2.      * by the ThreadLocal class. */ 
  3. ThreadLocal.ThreadLocalMap threadLocals = null
  4.  
  5.     /* 
  6.      * InheritableThreadLocal values pertaining to this thread. This map is 
  7.      * maintained by the InheritableThreadLocal class. 
  8.      */ 
  9. ThreadLocal.ThreadLocalMap inheritableThreadLocals = null

不要高興太早,對于使用線程池的情況,由于會緩存線程,線程是緩存起來反復使用的。這時父子線程關系的上下文傳遞,已經沒有意義。

附加問:你如何解決的?

阿里這里有個庫,https://github.com/alibaba/transmittable-thread-local 專門解決變量跨線程共享。如果你面的阿里,不妨順便舔一把。

3.4 Netty的FastThreadLocal是什么

既然Java中有了ThreadLocal類了,為什么Netty還自己創建了一個叫做FastThreadLocal的結構?

我們首先來看一下ThreadLocal的實現。

Thread類中,有一個成員變量threadLocals,存放了與本線程相關的所有自定義信息。對這個變量的定義在Thread類,而操作卻在ThreadLocal類中。

問題就出在ThreadLocalMap類上,它雖然叫Map,但卻沒有實現Map的接口。如圖,ThreadLocalMap在rehash的時候,并沒有采用類似HashMap的數組+鏈表+紅黑樹的做法,它只使用了一個數組,使用開放尋址(遇到沖突,依次查找,直到空閑位置)的方法,這種方式是非常低效的。

由于Netty對ThreadLocal的使用非常頻繁,Netty對它進行了專項的優化。它之所以快,是因為在底層數據結構上做了文章,使用常量下標對元素進行定位,而不是使用JDK默認的探測性算法。

底層的InternalThreadLocalMap對cacheline也做了相應的優化。

作者簡介:小姐姐味道 (xjjdog),一個不允許程序員走彎路的公眾號。聚焦基礎架構和Linux。十年架構,日百億流量,與你探討高并發世界,給你不一樣的味道。我的個人微信xjjdog0,歡迎添加好友,進一步交流。

 

責任編輯:武曉燕 來源: 小姐姐味道
相關推薦

2022-05-14 21:19:22

ThreadLocaJDKsynchroniz

2021-10-26 15:56:57

kafka數據平臺,

2024-03-13 13:56:11

openFeignHttp服務調用

2023-02-26 02:00:36

OpenFeign接口實現類

2021-07-21 09:15:27

MySQL數據庫面試

2022-04-01 12:40:13

MySQL數據庫

2022-01-24 14:08:16

Redis面試命令

2023-03-08 09:03:55

2023-04-26 09:16:17

2021-07-30 16:16:54

網絡面試TCP

2020-07-28 08:59:22

JavahreadLocal面試

2021-06-04 14:38:12

網絡通信TCP揮手

2021-08-27 14:14:39

ThreadLocal源碼操作

2022-03-25 09:01:36

Spring循環依賴面試

2022-11-04 08:47:52

底層算法數據

2021-11-08 14:10:37

分布式Spring鏈路

2019-05-29 15:17:43

TCPHTTPSSL

2023-11-03 08:10:49

ThreadLoca內存泄露

2024-10-28 08:15:32

2020-09-11 06:39:29

ThreadLocal線程
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 成人免费观看男女羞羞视频 | 亚洲一区二区三区在线 | 亚洲一区二区在线播放 | 久久综合九色综合欧美狠狠 | av官网在线 | 日韩在线欧美 | 免费一级片 | www.99re| 成人免费观看男女羞羞视频 | 久久精品女人天堂av | 99这里只有精品 | 天堂视频免费 | 精品av久久久久电影 | 久久久久久久国产精品视频 | 亚洲精选一区 | 亚洲精品91| 成人国产在线视频 | 日韩中文一区二区 | 久久一级免费视频 | 日本在线免费视频 | 免费观看黄色片视频 | 午夜一区 | 亚洲一区二区三区在线 | 国产精品成人在线播放 | 久久久久无码国产精品一区 | 国精产品一区一区三区免费完 | 国产欧美精品区一区二区三区 | 国产精品毛片一区二区在线看 | 一区二区三区四区在线 | 国产一区 在线视频 | 国产 日韩 欧美 在线 | 亚洲性视频网站 | 天堂成人国产精品一区 | 国产免费va| 国产电影精品久久 | 性在线 | 日本欧美国产在线观看 | 日产久久 | 亚洲综合久久久 | 色综合久久88色综合天天 | 国产日韩欧美一区二区在线播放 |