精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

細數ThreadLocal三大坑,內存泄露僅是小兒科

開發 后端
ThreadLocal寫錯難,但是用錯就很容易,本文將會詳細總結ThreadLocal容易用錯的三個坑。

 

[[398500]]

 我在參加Code Review的時候不止一次聽到有同學說:我寫的這個上下文工具沒問題,在線上跑了好久了。其實這種想法是有問題的,ThreadLocal寫錯難,但是用錯就很容易,本文將會詳細總結ThreadLocal容易用錯的三個坑:

內存泄露

線程池中線程上下文丟失

并行流中線程上下文丟失

內存泄露

由于ThreadLocal的key是弱引用,因此如果使用后不調用remove清理的話會導致對應的value內存泄露。 

  1. @Test  
  2. public void testThreadLocalMemoryLeaks() {  
  3.     ThreadLocal<List<Integer>> localCache = new ThreadLocal<>();  
  4.    List<Integer> cacheInstance = new ArrayList<>(10000);  
  5.     localCache.set(cacheInstance);  
  6.     localCache = new ThreadLocal<>();  

當localCache的值被重置之后cacheInstance被ThreadLocalMap中的value引用,無法被GC,但是其key對ThreadLocal實例的引用是一個弱引用,本來ThreadLocal的實例被localCache和ThreadLocalMap的key同時引用,但是當localCache的引用被重置之后,則ThreadLocal的實例只有ThreadLocalMap的key這樣一個弱引用了,此時這個實例在GC的時候能夠被清理。

其實看過ThreadLocal源碼的同學會知道,ThreadLocal本身對于key為null的Entity有自清理的過程,但是這個過程是依賴于后續對ThreadLocal的繼續使用,假如上面的這段代碼是處于一個秒殺場景下,會有一個瞬間的流量峰值,這個流量峰值也會將集群的內存打到高位(或者運氣不好的話直接將集群內存打滿導致故障),后面由于峰值流量已過,對ThreadLocal的調用也下降,會使得ThreadLocal的自清理能力下降,造成內存泄露。ThreadLocal的自清理是錦上添花,千萬不要指望他雪中送碳。

相比于ThreadLocal中存儲的value對象泄露,ThreadLocal用在web容器中時更需要注意其引起的ClassLoader泄露。

Tomcat官網對在web容器中使用ThreadLocal引起的內存泄露做了一個總結,詳見:https://cwiki.apache.org/confluence/display/tomcat/MemoryLeakProtection,這里我們列舉其中的一個例子。

熟悉Tomcat的同學知道,Tomcat中的web應用由Webapp Classloader這個類加載器的,并且Webapp Classloader是破壞雙親委派機制實現的,即所有的web應用先由Webapp classloader加載,這樣的好處就是可以讓同一個容器中的web應用以及依賴隔離。

下面我們看具體的內存泄露的例子: 

  1. public class MyCounter {  
  2.  private int count = 0 
  3.  public void increment() {  
  4.   count++;  
  5.  }  
  6.  public int getCount() {  
  7.   return count;  
  8.  }  
  9.  
  10. public class MyThreadLocal extends ThreadLocal<MyCounter> {  
  11.  
  12. public class LeakingServlet extends HttpServlet {  
  13.  private static MyThreadLocal myThreadLocal = new MyThreadLocal();  
  14.  protected void doGet(HttpServletRequest request,  
  15.    HttpServletResponse response) throws ServletException, IOException {  
  16.   MyCounter counter = myThreadLocal.get();  
  17.   if (counter == null) {  
  18.    counter = new MyCounter();  
  19.    myThreadLocal.set(counter);  
  20.   }  
  21.   response.getWriter().println(  
  22.     "The current thread served this servlet " + counter.getCount()  
  23.       + " times"); 
  24.    counter.increment();  
  25.  }  

需要注意這個例子中的兩個非常關鍵的點:

MyCounter以及MyThreadLocal必須放到web應用的路徑中,保被Webapp Classloader加載

ThreadLocal類一定得是ThreadLocal的繼承類,比如例子中的MyThreadLocal,因為ThreadLocal本來被Common Classloader加載,其生命周期與Tomcat容器一致。ThreadLocal的繼承類包括比較常見的NamedThreadLocal,注意不要踩坑。

假如LeakingServlet所在的Web應用啟動,MyThreadLocal類也會被Webapp Classloader加載,如果此時web應用下線,而線程的生命周期未結束(比如為LeakingServlet提供服務的線程是一個線程池中的線程),那會導致myThreadLocal的實例仍然被這個線程引用,而不能被GC,期初看來這個帶來的問題也不大,因為myThreadLocal所引用的對象占用的內存空間不太多,問題在于myThreadLocal間接持有加載web應用的webapp classloader的引用(通過myThreadLocal.getClass().getClassLoader()可以引用到),而加載web應用的webapp classloader有持有它加載的所有類的引用,這就引起了Classloader泄露,它泄露的內存就非常可觀了。

線程池中線程上下文丟失

ThreadLocal不能在父子線程中傳遞,因此最常見的做法是把父線程中的ThreadLocal值拷貝到子線程中,因此大家會經常看到類似下面的這段代碼: 

  1. for(value in valueList){  
  2.      Future<?> taskResult = threadPool.submit(new BizTask(ContextHolder.get()));//提交任務,并設置拷貝Context到子線程  
  3.      results.add(taskResult);  
  4.  
  5. for(result in results){  
  6.     result.get();//阻塞等待任務執行完成  

提交的任務定義長這樣: 

  1. class BizTask<T> implements Callable<T>  {  
  2.     private String session = null;     
  3.     public BizTask(String session) {  
  4.         this.session = session;  
  5.     }      
  6.     @Override  
  7.     public T call(){ 
  8.          try {  
  9.             ContextHolder.set(this.session);  
  10.             // 執行業務邏輯  
  11.         } catch(Exception e){  
  12.             //log error  
  13.         } finally {  
  14.             ContextHolder.remove(); // 清理 ThreadLocal 的上下文,避免線程復用時context互串  
  15.         }  
  16.         return null;  
  17.     }  

對應的線程上下文管理類為: 

  1. class ContextHolder {  
  2.     private static ThreadLocal<String> localThreadCache = new ThreadLocal<>();     
  3.     public static void set(String cacheValue) {  
  4.         localThreadCache.set(cacheValue); 
  5.     }      
  6.     public static String get() {  
  7.         return localThreadCache.get();  
  8.     }      
  9.     public static void remove() {  
  10.         localThreadCache.remove();  
  11.     }     

這么寫倒也沒有問題,我們再看看線程池的設置: 

  1. ThreadPoolExecutor executorPool = new ThreadPoolExecutor(20, 40, 30, TimeUnit.SECONDS, new LinkedBlockingQueue<Runnable>(40), new XXXThreadFactory(), ThreadPoolExecutor.CallerRunsPolicy); 

其中最后一個參數控制著當線程池滿時,該如何處理提交的任務,內置有4種策略 

  1. ThreadPoolExecutor.AbortPolicy //直接拋出異常  
  2. ThreadPoolExecutor.DiscardPolicy //丟棄當前任務  
  3. ThreadPoolExecutor.DiscardOldestPolicy //丟棄工作隊列頭部的任務  
  4. ThreadPoolExecutor.CallerRunsPolicy //轉串行執行 

可以看到,我們初始化線程池的時候指定如果線程池滿,則新提交的任務轉為串行執行,那我們之前的寫法就會有問題了,串行執行的時候調用ContextHolder.remove();會將主線程的上下文也清理,即使后面線程池繼續并行工作,傳給子線程的上下文也已經是null了,而且這樣的問題很難在預發測試的時候發現。

并行流中線程上下文丟失

如果ThreadLocal碰到并行流,也會有很多有意思的事情發生,比如有下面的代碼: 

  1. class ParallelProcessor<T> {   
  2.     public void process(List<T> dataList) {  
  3.         // 先校驗參數,篇幅限制先省略不寫  
  4.         dataList.parallelStream().forEach(entry -> {  
  5.             doIt();  
  6.         }); 
  7.     } 
  8.      private void doIt() {  
  9.         String session = ContextHolder.get();  
  10.         // do something  
  11.     }  

這段代碼很容易在線下測試的過程中發現不能按照預期工作,因為并行流底層的實現也是一個ForkJoin線程池,既然是線程池,那ContextHolder.get()可能取出來的就是一個null。我們順著這個思路把代碼再改一下: 

  1. class ParallelProcessor<T> {   
  2.      private String session;     
  3.     public ParallelProcessor(String session) {  
  4.         this.session = session;  
  5.     }      
  6.     public void process(List<T> dataList) {  
  7.         // 先校驗參數,篇幅限制先省略不寫  
  8.         dataList.parallelStream().forEach(entry -> {  
  9.             try {  
  10.                 ContextHolder.set(session);  
  11.                 // 業務處理  
  12.                 doIt();  
  13.             } catch (Exception e) {  
  14.                 // log it  
  15.             } finally {  
  16.                 ContextHolder.remove();  
  17.             }  
  18.         });  
  19.     }  
  20.      private void doIt() {  
  21.         String session = ContextHolder.get();  
  22.         // do something  
  23.     }  

修改完后的這段代碼可以工作嗎?如果運氣好,你會發現這樣改又有問題,運氣不好,這段代碼在線下運行良好,這段代碼就順利上線了。不久你就會發現系統中會有一些其他很詭異的bug。原因在于并行流的設計比較特殊,父線程也有可能參與到并行流線程池的調度,那如果上面的process方法被父線程執行,那么父線程的上下文會被清理。導致后續拷貝到子線程的上下文都為null,同樣產生丟失上下文的問題,關于并行流的實現可以參考文章啥?用了并行流還更慢了。 

 

責任編輯:龐桂玉 來源: Java知音
相關推薦

2022-03-17 17:54:19

人臉識別AI人工智能

2010-01-07 15:07:34

Ubuntu Anju

2022-06-06 00:25:09

Golangpanic死鎖

2022-08-26 07:33:49

內存JVMEntry

2025-11-13 08:22:03

2024-10-31 09:24:42

2019-08-07 06:16:28

物聯網IOT技術

2021-12-28 00:27:24

運營商內卷網絡

2023-06-30 08:10:14

JavaBigDecimal

2023-04-16 19:34:01

2022-10-18 08:38:16

內存泄漏線程

2018-04-02 07:32:15

2023-02-17 08:20:24

SQL腳本數據庫

2017-03-30 08:42:42

技術信息安全開源

2021-04-23 20:59:02

ThreadLocal內存

2021-09-26 09:16:45

RedisGeo 類型數據類型

2024-05-30 12:15:04

2015-11-20 10:39:31

2018-07-19 16:00:25

2022-05-15 08:13:50

Mysql數據庫Mycat
點贊
收藏

51CTO技術棧公眾號

欧美激情五月| 日本妇乱大交xxxxx| 日韩精品无码一区二区三区| 中文字幕av久久爽一区| 午夜不卡一区| 亚洲国产成人va在线观看天堂| 黑人巨大精品欧美一区二区小视频 | 极品美鲍一区| 国产免费叼嘿网站免费| 国产成人手机高清在线观看网站| 欧美综合久久久| 人妻无码一区二区三区四区| 清纯唯美亚洲色图| 国产一区二区三区免费看| 97视频免费在线观看| 一本一本久久a久久| 噜噜噜天天躁狠狠躁夜夜精品| 欧美在线免费观看亚洲| 国产日韩av网站| 麻豆视频网站在线观看| 成人午夜激情在线| 国产日韩在线免费| 国产一级精品视频| 欧美1区2区3区| 中文字幕亚洲二区| 亚洲调教欧美在线| 视频成人永久免费视频| 欧美影片第一页| 久久成人免费观看| 日韩av毛片| 亚洲天堂2016| 五月天久久综合网| 男女视频在线观看免费| 国产成人精品午夜视频免费| 成人国产精品一区| 午夜视频网站在线观看| 国产午夜精品一区二区三区欧美| 欧美成人性生活| av资源在线免费观看| 中文字幕中文字幕精品| 日韩www在线| 国产草草浮力影院| 97超碰成人| 欧美日韩大陆在线| 手机av在线网| 成人全视频免费观看在线看| 在线亚洲欧美专区二区| 大肉大捧一进一出好爽动态图| 擼擼色在线看观看免费| 亚洲一区二区三区四区五区中文| 一级黄色片播放| 国产不卡在线| 亚洲精品久久久蜜桃| 热这里只有精品| 免费a级在线播放| 国产精品久久久久久亚洲伦| 亚洲国产一区二区精品视频| jizzjizz在线观看| 国产精品美女久久福利网站| 亚洲一区二区三区在线观看视频| av在线免费观看网| 国产精品初高中害羞小美女文| 视频一区视频二区视频三区视频四区国产 | 一二区成人影院电影网| 在线观看日韩av先锋影音电影院| 久久久精品三级| 欧美美女福利视频| 欧美一区2区视频在线观看| 男人操女人下面视频| 99久久免费精品国产72精品九九| 欧美v亚洲v综合ⅴ国产v| 国产人成视频在线观看| 日本福利一区| 一个色综合导航| 久久高清内射无套| 欧美三级免费| 日本免费久久高清视频| 波多野结衣视频免费观看| 久久精品国产99| 欧美影视一区二区三区| 国产剧情日韩欧美| 国产精品久久影视| 成人午夜电影久久影院| 明星裸体视频一区二区| 秋霞午夜在线观看| 亚洲小说欧美激情另类| 国产视频一区二区三区在线播放| 国产精品久久久久久久久久齐齐| 91精品国产一区二区三区香蕉| 美女流白浆视频| 天堂99x99es久久精品免费| 在线观看国产精品91| 波多野结衣不卡视频| 国产亚洲一级| 国产精品视频一区二区三区四| 精品国产伦一区二区三区| www.在线欧美| 一区精品视频| 男人天堂视频在线观看| 欧美色涩在线第一页| 亚洲成年人在线观看| 成人情趣视频| 韩国v欧美v日本v亚洲| 中文在线a天堂| 成人一区二区视频| 亚洲欧美丝袜| 天堂√中文最新版在线| 欧美精品乱人伦久久久久久| 人妻少妇精品视频一区二区三区 | 亚洲精选中文字幕| 中文字幕在线2021| 日韩黄色一级片| 国产精品theporn88| 欧美黑人激情| 黑人精品xxx一区一二区| 99日在线视频| 国产影视一区| 97热在线精品视频在线观看| 国产精品福利电影| 国产亚洲精品福利| 免费一级特黄特色毛片久久看| 日韩专区视频| 国产一区二区三区网站| 日本三级网站在线观看| 国产麻豆日韩欧美久久| 一区二区不卡视频| 亚洲成人不卡| 亚洲欧美国内爽妇网| 国产无遮挡aaa片爽爽| 国内精品免费**视频| 亚洲aⅴ天堂av在线电影软件| 男女羞羞在线观看| 亚洲国内精品在线| 国产一级免费观看| 国产成人av资源| 午夜久久久久久久久久久| 欧美日韩国产v| 日韩精品免费在线视频| 精品无码久久久久久久| 国内精品免费**视频| 亚洲欧美日韩精品在线| 97精品国产99久久久久久免费| 日韩精品在线影院| 成人免费区一区二区三区| 国产99一区视频免费| 日本黄网站色大片免费观看| av日韩一区| 久久国产精品久久久| 国产美女www爽爽爽视频| 国产精品国产精品国产专区不蜜| 亚洲欧美久久久久| 色中色综合网| 91免费国产视频| 大地资源网3页在线观看| 91精品麻豆日日躁夜夜躁| 国产探花在线免费观看| 国产一区 二区 三区一级| 欧美日韩亚洲国产成人| www久久久| 欧美高清videos高潮hd| 国 产 黄 色 大 片| 午夜影院久久久| 美女又爽又黄视频毛茸茸| 久久经典综合| 四虎影视永久免费在线观看一区二区三区| 日本成人片在线| 日韩在线欧美在线国产在线| 国产精品久久久久久久免费| 亚洲综合偷拍欧美一区色| 日本不卡视频一区| 欧美综合国产| 亚洲三区在线观看| 久久伊人影院| 91精品国产91久久| 国产高清在线| 在线播放日韩导航| 免费一级特黄特色大片| 26uuu精品一区二区| 香港日本韩国三级网站| 久久久精品久久久久久96| 粉嫩av免费一区二区三区| 少妇视频一区| 日韩中文字幕在线免费观看| 91麻豆成人精品国产免费网站| 一区二区三区在线免费播放| 国产精品边吃奶边做爽| 青草国产精品久久久久久| 欧美另类videosbestsex日本| 盗摄系列偷拍视频精品tp| 国产成人免费av电影| 国产网友自拍视频导航网站在线观看| 精品三级av在线| 一级黄色av片| 亚洲综合在线第一页| 欧美做受高潮6| 国产福利一区二区三区在线视频| av免费中文字幕| 羞羞色午夜精品一区二区三区| 国内一区二区在线视频观看| 欧美一级免费| 国产69久久精品成人看| 国产人成网在线播放va免费| 亚洲精品美女视频| 国产内射老熟女aaaa∵| 欧美日韩视频在线| 少妇人妻丰满做爰xxx| 99精品国产91久久久久久| 亚洲黄色av片| 天堂影院一区二区| 9色porny| 天天综合久久| 免费亚洲精品视频| 国产精品调教视频| 91精品视频一区| 亚洲mmav| 2019中文字幕全在线观看| aaa大片在线观看| 在线电影中文日韩| 日韩欧美电影在线观看| 欧美v亚洲v综合ⅴ国产v| 国产精品伦理一区| 欧美午夜精品久久久| 日日摸天天添天天添破| 亚洲综合男人的天堂| 亚洲伦理一区二区三区| 国产亚洲va综合人人澡精品| 欧美一级片黄色| 国产成都精品91一区二区三| 看看黄色一级片| 美女在线视频一区| 日本黄网站免费| 亚洲一区区二区| 欧美激情视频免费看| 欧美日韩 国产精品| 国产精品亚洲天堂| 日韩电影在线视频| 日韩和欧美的一区二区| 伊人久久大香线蕉av不卡| 狠狠爱一区二区三区| 国产厕拍一区| 国产精品亚洲一区| 国产精品17p| 国产欧美丝袜| 欧美freesex8一10精品| 国内精品久久久久久久果冻传媒| 成人av动漫| 国产欧美日本在线| 美女福利一区| 免费99视频| 国产精品一区高清| 日韩欧美精品久久| 欧美综合一区| 亚洲最大色综合成人av| 国产精品久久观看| 五月天综合婷婷| 欧美日韩国产色综合一二三四| 亚洲免费视频播放| 亚洲午夜91| 免费看国产一级片| 日韩电影网1区2区| 亚洲久久中文字幕| 国产在线视频一区二区三区| 妖精视频在线观看| 国产盗摄一区二区三区| www.com日本| 95精品视频在线| 久久久久久久久久久久| 日本一区二区视频在线观看| 欧日韩不卡视频| 尤物在线观看一区| 可以免费看的av毛片| 色偷偷久久人人79超碰人人澡| 亚洲精品久久久久久久蜜桃| 欧美男同性恋视频网站| 成人午夜福利视频| 日韩久久免费视频| 91网页在线观看| 欧美黑人国产人伦爽爽爽| 国产精品蜜芽在线观看| 国产精品视频26uuu| 国产高清日韩| 国产亚洲二区| 日产午夜精品一线二线三线| 中文字幕精品在线播放| 国产精品嫩草99av在线| 三区视频在线观看| www.欧美日韩国产在线| 五月婷婷欧美激情| 亚洲一区成人在线| 国产精品成人久久久| 日韩欧美www| 黄色小视频在线观看| 久久综合伊人77777尤物| 蜜桃视频m3u8在线观看| 国产精品一区=区| 精品嫩草影院| 亚洲一一在线| 国产亚洲毛片| 色诱av手机版| 国产精品区一区二区三区| 国产在线视频卡一卡二| 欧美日韩一区二区在线观看视频| 亚洲av永久无码国产精品久久 | 俄罗斯精品一区二区| 精品国产精品国产偷麻豆| 日韩一级特黄毛片| 日韩影院精彩在线| 国产精品扒开腿做爽爽爽a片唱戏| 国产农村妇女毛片精品久久麻豆 | 欧美疯狂性受xxxxx喷水图片| 免费av一级片| 久久久精品在线观看| 欧美舌奴丨vk视频| 国产视频一区二区不卡| 99精品全国免费观看视频软件| 国产中文字幕在线免费观看| 国产最新精品精品你懂的| 91网站免费入口| 亚洲国产欧美日韩另类综合| 国产精品视频第一页| 亚洲欧美国产视频| 欧美日韩国产观看视频| 99国产超薄丝袜足j在线观看| 成人网18免费网站| 国产日韩一区二区在线| 成人精品国产一区二区4080| 久草福利资源在线| 色婷婷av一区二区三区大白胸| 天堂成人在线视频| 欧美极品欧美精品欧美视频| gogo大尺度成人免费视频| 亚洲欧美日韩精品在线| 日韩电影网1区2区| 天堂在线中文视频| 色88888久久久久久影院野外| 亚洲欧美日韩综合在线| 久久久欧美精品| 成人搞黄视频| 日本a在线天堂| 国产凹凸在线观看一区二区| 久久国产美女视频| 欧美一二三四在线| 成人在线网址| 亚洲综合社区网| 在线看片不卡| 国产精品19p| 亚洲一区二区三区四区五区黄| 午夜久久久久久久久久| 久久福利网址导航| 久久精品免视看国产成人| 青青视频免费在线| 国产精品一品视频| 久久久一区二区三区四区| 精品99一区二区三区| 成人性生交大片免费看在线播放| 成人毛片网站| 亚洲精品孕妇| 久久精品一区二区免费播放| 色综合久久久久久久久久久| 超碰免费在线| 96pao国产成视频永久免费| 在线观看国产精品入口| 亚洲精品第二页| 欧美午夜精品伦理| a天堂在线资源| 91精品在线观看视频| 欧美日韩视频| 精品中文字幕在线播放| 在线免费视频一区二区| 亚洲搞黄视频| www日韩av| 国产午夜久久| www.4hu95.com四虎| 日韩亚洲欧美一区二区三区| 91精品国产黑色瑜伽裤| 日本一区二区三区四区高清视频 | 成人春色激情网| 欧美体内she精视频在线观看| 国产草草浮力影院| 欧美三级蜜桃2在线观看| 性xxxxfjsxxxxx欧美| 国产一区二区免费在线观看| 久久久久网站| 欧美日韩中文字幕在线观看 | 麻豆传媒视频在线观看| 日韩国产欧美一区二区三区| 亚洲不卡的av| 欧美精品一区二区三| 欧美成a人片在线观看久| 免费在线精品视频| www一区二区| a毛片在线免费观看| 国产91精品网站| 欧美日本一区二区视频在线观看 | 哺乳一区二区三区中文视频 | 国产一区二区三区在线观看免费视频| 国产精品不卡av| www国产亚洲精品久久网站| 黄色欧美在线| 午夜激情影院在线观看|