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

驚!ThreadLocal你怎么動不動就內存泄漏?

存儲 存儲軟件
使用 ThreadLocal 不當可能會導致內存泄露,是什么原因導致的內存泄漏呢?

[[382111]] 

本文轉載自微信公眾號「無聊學Java」,作者無聊  。轉載本文請聯系無聊學Java公眾號。 

今天無聊帶大家分析下ThreadLocal為什么會內存泄漏~

前言

使用 ThreadLocal 不當可能會導致內存泄露,是什么原因導致的內存泄漏呢?

正文

我們首先看一個例子,代碼如下:

  1. public class ThreadLocalOutOfMemoryTest { 
  2.     static class LocalVariable { 
  3.         private Long[] a = new Long[1024*1024]; 
  4.     } 
  5.  
  6.     // (1) 
  7.     final static ThreadPoolExecutor poolExecutor = new ThreadPoolExecutor(6, 6, 1, TimeUnit.MINUTES, 
  8.             new LinkedBlockingQueue<>()); 
  9.     // (2) 
  10.     final static ThreadLocal<LocalVariable> localVariable = new ThreadLocal<LocalVariable>(); 
  11.  
  12.     public static void main(String[] args) throws InterruptedException { 
  13.         // (3) 
  14.         for (int i = 0; i < 50; ++i) { 
  15.             poolExecutor.execute(new Runnable() { 
  16.                 public void run() { 
  17.                     // (4) 
  18.                     localVariable.set(new LocalVariable()); 
  19.                     // (5) 
  20.                     System.out.println("use local varaible"); 
  21. //                    localVariable.remove(); 
  22.  
  23.                 } 
  24.             }); 
  25.  
  26.             Thread.sleep(1000); 
  27.         } 
  28.         // (6) 
  29.         System.out.println("pool execute over"); 
  30.     } 

代碼(1)創建了一個核心線程數和最大線程數為 6 的線程池,這個保證了線程池里面隨時都有 6 個線程在運行。

代碼(2)創建了一個 ThreadLocal 的變量,泛型參數為 LocalVariable,LocalVariable 內部是一個 Long 數組。

代碼(3)向線程池里面放入 50 個任務。

代碼(4)設置當前線程的 localVariable 變量,也就是把 new 的 LocalVariable 變量放入當前線程的 threadLocals 變量。

由于沒有調用線程池的 shutdown 或者 shutdownNow 方法所以線程池里面的用戶線程不會退出,進而 JVM 進程也不會退出。

運行后,我們立即打開jconsole 監控堆內存變化,如下圖:

接著,讓我們打開 localVariable.remove() 注釋,然后在運行,觀察堆內存變化如下:

從第一次運行結果可知,當主線程處于休眠時候進程占用了大概 75M 內存,打開 localVariable.remove() 注釋后第二次運行則占用了大概 25M 內存,可知 沒有寫 localVariable.remove() 時候內存發生了泄露,下面分析下泄露的原因,如下:

“第一次運行的代碼,在設置線程的 localVariable 變量后沒有調用localVariable.remove() 方法,導致線程池里面的 5 個線程的 threadLocals 變量里面的new LocalVariable()實例沒有被釋放,雖然線程池里面的任務執行完畢了,但是線程池里面的 5 個線程會一直存在直到 JVM 退出。這里需要注意的是由于 localVariable 被聲明了 static,雖然線程的 ThreadLocalMap 里面是對 localVariable 的弱引用,localVariable 也不會被回收。運行結果二的代碼由于線程在設置 localVariable 變量后即使調用了localVariable.remove()方法進行了清理,所以不會存在內存泄露。

接下來我們要想清楚的知道內存泄漏的根本原因,那么我們就要進入源碼去看了。

我們知道ThreadLocal 只是一個工具類,具體存放變量的是在線程的 threadLocals 變量里面,threadLocals 是一個 ThreadLocalMap 類型的,我們首先一覽ThreadLocalMap的類圖結構,類圖結構如下圖:

如上圖 ThreadLocalMap 內部是一個 Entry 數組, Entry 繼承自 WeakReference,Entry 內部的 value 用來存放通過 ThreadLocal 的 set 方法傳遞的值,那么 ThreadLocal 對象本身存放到哪里了嗎?

下面看看 Entry 的構造函數,如下所示:

  1. Entry(ThreadLocal<?> k, Object v) { 
  2.     super(k); 
  3.     value = v; 

接著我們再接著看Entry的父類WeakReference的構造函數super(k),如下所示:

  1. public WeakReference(T referent) { 
  2.    super(referent); 

接著我們再看WeakReference的父類Reference的構造函數super(referent),如下所示:

  1. Reference(T referent) { 
  2.    this(referent, null); 

接著我們再看WeakReference的父類Reference的另外一個構造函數this(referent , null),如下所示:

  1. Reference(T referent, ReferenceQueue<? super T> queue) { 
  2.    this.referent = referent; 
  3.    this.queue = (queue == null) ? ReferenceQueue.NULL : queue; 

可知 k 被傳遞到了 WeakReference 的構造函數里面,也就是說 ThreadLocalMap 里面的 key 為 ThreadLocal 對象的弱引用,具體是 referent 變量引用了 ThreadLocal 對象,value 為具體調用 ThreadLocal 的 set 方法傳遞的值。

當一個線程調用 ThreadLocal 的 set 方法設置變量時候,當前線程的 ThreadLocalMap 里面就會存放一個記錄,這個記錄的 key 為 ThreadLocal 的引用,value 則為設置的值。

但是考慮如果這個 ThreadLocal 變量沒有了其他強依賴,而當前線程還存在的情況下,由于線程的 ThreadLocalMap 里面的 key 是弱依賴,則當前線程的 ThreadLocalMap 里面的 ThreadLocal 變量的弱引用會被在 gc 的時候回收,但是對應 value 還是會造成內存泄露,這時候 ThreadLocalMap 里面就會存在 key 為 null 但是 value 不為 null 的 entry 項。

其實在 ThreadLocal 的 set 和 get 和 remove 方法里面有一些時機是會對這些 key 為 null 的 entry 進行清理的,但是這些清理不是必須發生的,下面簡單講解ThreadLocalMap 的 remove 方法的清理過程,remove 的源碼,如下所示:

  1. private void remove(ThreadLocal<?> key) { 
  2.  
  3.   //(1)計算當前ThreadLocal變量所在table數組位置,嘗試使用快速定位方法 
  4.   Entry[] tab = table
  5.   int len = tab.length; 
  6.   int i = key.threadLocalHashCode & (len-1); 
  7.   //(2)這里使用循環是防止快速定位失效后,變量table數組 
  8.   for (Entry e = tab[i]; e != null; e = tab[i = nextIndex(i, len)]) { 
  9.       //(3)找到 
  10.       if (e.get() == key) { 
  11.           //(4)找到則調用WeakReference的clear方法清除對ThreadLocal的弱引用 
  12.           e.clear(); 
  13.           //(5)清理keynull的元素 
  14.           expungeStaleEntry(i); 
  15.           return
  16.       } 
  17.    } 
  18.  private int expungeStaleEntry(int staleSlot) { 
  19.             Entry[] tab = table
  20.             int len = tab.length; 
  21.             //(6)去掉去value的引用 
  22.             tab[staleSlot].value = null
  23.             tab[staleSlot] = null
  24.             size--; 
  25.             Entry e; 
  26.             int i; 
  27.             for (i = nextIndex(staleSlot, len); (e = tab[i]) != null; i = nextIndex(i, len)) { 
  28.                 ThreadLocal<?> k = e.get(); 
  29.                 //(7)如果keynull,則去掉對value的引用。 
  30.                 if (k == null) { 
  31.                     e.value = null
  32.                     tab[i] = null
  33.                     size--; 
  34.                 } else { 
  35.                     int h = k.threadLocalHashCode & (len - 1); 
  36.                     if (h != i) { 
  37.                         tab[i] = null
  38.                         while (tab[h] != null
  39.                             h = nextIndex(h, len); 
  40.                         tab[h] = e; 
  41.                     } 
  42.                 } 
  43.             } 
  44.             return i; 
  45.   } 

代碼(4)調用了 Entry 的 clear 方法,實際調用的是父類 WeakReference 的 clear 方法,作用是去掉對 ThreadLocal 的弱引用。

代碼(6)是去掉對 value 的引用,到這里當前線程里面的當前 ThreadLocal 對象的信息被清理完畢了。

代碼(7)從當前元素的下標開始看 table 數組里面的其他元素是否有 key 為 null 的,有則清理。循環退出的條件是遇到 table 里面有 null 的元素。所以這里知道 null 元素后面的 Entry 里面 key 為 null 的元素不會被清理。

總結

  1. ThreadLocalMap 內部 Entry 中 key 使用的是對 ThreadLocal 對象的弱引用,這為避免內存泄露是一個進步,因為如果是強引用,那么即使其他地方沒有對 ThreadLocal 對象的引用,ThreadLocalMap 中的 ThreadLocal 對象還是不會被回收,而如果是弱引用則這時候 ThreadLocal 引用是會被回收掉的。
  2. 但是對于的 value 還是不能被回收,這時候 ThreadLocalMap 里面就會存在 key 為 null 但是 value 不為 null 的 entry 項,雖然 ThreadLocalMap 提供了 set,get,remove 方法在一些時機下會對這些 Entry 項進行清理,但是這是不及時的,也不是每次都會執行的,所以一些情況下還是會發生內存泄露,所以在使用完畢后即使調用 remove 方法才是解決內存泄露的最好辦法。
  3. 線程池里面設置了 ThreadLocal 變量一定要記得及時清理,因為線程池里面的核心線程是一直存在的,如果不清理,那么線程池的核心線程的 threadLocals 變量一直會持有 ThreadLocal 變量。

 

 

責任編輯:武曉燕 來源: 無聊學Java
相關推薦

2020-09-10 07:40:28

ThreadLocal內存

2020-09-11 07:38:50

內存泄漏檢測

2020-11-09 06:00:04

Windows 10Windows操作系統

2021-08-10 09:58:59

ThreadLocal內存泄漏

2018-10-25 15:24:10

ThreadLocal內存泄漏Java

2022-05-09 14:09:23

多線程線程安全

2024-03-22 13:31:00

線程策略線程池

2025-04-01 05:22:00

JavaThread變量

2024-09-29 08:57:25

2022-11-04 08:47:52

底層算法數據

2024-02-02 09:00:14

內存泄漏對象

2025-10-27 01:33:00

2018-05-08 15:42:30

PC升級筆記本

2024-03-11 08:22:40

Java內存泄漏

2023-12-18 10:45:23

內存泄漏計算機服務器

2012-02-22 21:28:58

內存泄漏

2025-03-28 08:53:51

2022-08-26 07:33:49

內存JVMEntry

2015-03-30 11:18:50

內存管理Android

2023-11-28 12:25:02

多線程安全
點贊
收藏

51CTO技術棧公眾號

动漫精品一区二区| 成人18精品视频| 久久久精品一区二区| 国产免费无码一区二区| 成人性生活视频| 亚洲欧美一区二区视频| 国语精品免费视频| 国产精品高潮呻吟av| 亚洲精品欧洲| 久久精品国产亚洲精品| 亚洲永久无码7777kkk| 日韩免费在线电影| 一本色道久久综合狠狠躁的推荐 | 午夜精品久久久久久久久久久久久| 无码人妻精品一区二区三区温州| 伊人久久大香伊蕉在人线观看热v| 亚洲成人午夜影院| 在线观看三级网站| 岛国最新视频免费在线观看| 国产成都精品91一区二区三| 国产精品久久久91| 亚洲精品男人天堂| 精久久久久久| 久久高清视频免费| 国产无遮挡在线观看| 网曝91综合精品门事件在线| 日韩午夜激情电影| 不卡的av中文字幕| 日韩久久一区二区三区| 午夜精品福利久久久| 日韩精品第1页| 伊人免费在线| 国产人久久人人人人爽| 精品无人乱码一区二区三区的优势 | 国产一二三在线观看| bt7086福利一区国产| aaa级精品久久久国产片| 中文字幕免费在线看| 久久xxxx| 日本人成精品视频在线| 日韩精品成人在线| 影音先锋一区| 久久人人爽国产| 久久久久亚洲av成人片| 午夜日韩av| 美女av一区二区| 日本视频在线免费| 久久日文中文字幕乱码| 中文字幕亚洲欧美日韩2019| 中文字幕在线看高清电影| 欧美美女在线直播| 亚洲国产一区二区三区四区| 日本少妇一区二区三区| 国产日韩中文在线中文字幕| 67194成人在线观看| www.久久久久久久久久久| 久久影视精品| 3d成人h动漫网站入口| 在线观看免费av网址| 91麻豆精品| 欧美一区二区在线免费观看| 能看毛片的网站| 中文字幕日韩高清在线| 精品福利一区二区三区| av2014天堂网| 欧美亚洲国产一区| 日韩在线中文字| 久久国产美女视频| 国产综合视频| 欧美一区二区三区……| 无码人妻一区二区三区免费| 蜜桃av一区二区| 91久久精品视频| 亚洲第一色视频| 99re这里只有精品首页| 日本一区视频在线观看| 欧美人xxx| 亚洲国产精品人人做人人爽| www一区二区www免费| 日韩欧美少妇| 日韩欧美国产一区在线观看| 免费成人蒂法网站| 欧美一区二区三| 久久综合免费视频| 欧美精品99久久久| 麻豆精品网站| 成人午夜激情免费视频| 蜜桃在线一区二区| 国产婷婷色一区二区三区四区| 在线观看成人一级片| bl视频在线免费观看| 欧美亚州韩日在线看免费版国语版| 三年中文在线观看免费大全中国| 波多野结衣在线一区二区| 亚洲深夜福利网站| 久久久久无码精品国产| 免费在线观看不卡| 国产高清在线精品一区二区三区| 噜噜噜噜噜在线视频| 亚洲精品免费电影| 亚洲三级视频网站| 国产精品高潮呻吟久久久久| 亚洲人a成www在线影院| 九九久久免费视频| 另类专区欧美蜜桃臀第一页| 国产精品一 二 三| 黄色一级大片在线免费看产| 色婷婷久久久亚洲一区二区三区| 波多野结衣免费观看| 国产精品美女久久久久久不卡| 色综合久久88| 97国产精品久久久| 国产人成亚洲第一网站在线播放| 青青草精品视频在线| 99久久这里有精品| 伊人久久男人天堂| 日韩女同强女同hd| 国产综合色在线| 午夜精品短视频| 中文日产幕无线码一区二区| 欧美大片在线观看一区二区| 日韩一区二区三区四区视频| 模特精品在线| 九色综合婷婷综合| av2020不卡| 日韩欧美国产一二三区| 神马久久精品综合| 精品在线免费观看| 亚洲 国产 日韩 综合一区| 亚洲天堂av影院| 亚洲第一视频网| 青青草国产在线观看| 另类的小说在线视频另类成人小视频在线 | 五月婷婷亚洲综合| 成+人+亚洲+综合天堂| 亚洲爆乳无码精品aaa片蜜桃| 福利一区二区| 色偷偷91综合久久噜噜| www.久久视频| 国产视频一区二区三区在线观看| 国产亚洲欧美在线视频| 麻豆一区二区麻豆免费观看| 国模精品一区二区三区色天香| 99久久久久久久| 亚洲精品乱码久久久久久黑人| 涩涩网站在线看| 中文字幕一区二区三三| **亚洲第一综合导航网站| 尤物视频在线看| 日韩精品一区二| 国产午夜小视频| 99久久综合99久久综合网站| 鲁一鲁一鲁一鲁一色| 色天天色综合| 国产精品 欧美在线| 国产粉嫩一区二区三区在线观看 | 992tv人人草| 亚洲成人精品| 97av自拍| 鲁鲁在线中文| 永久免费毛片在线播放不卡| 中文字幕观看视频| 亚洲免费在线播放| 亚洲麻豆一区二区三区| 一本色道88久久加勒比精品| 欧美一区二区三区四区在线观看地址 | 精品日本12videosex| 国产精品夜间视频香蕉| а天堂中文在线官网| 亚洲国产欧美一区| 自拍偷拍18p| 亚洲视频一区在线| 在线播放av网址| 乱码第一页成人| youjizz.com亚洲| 给我免费播放日韩视频| 国产99在线|中文| 免费观看久久久久| 精品少妇一区二区三区日产乱码| 日韩精品――中文字幕| 久久蜜桃av一区精品变态类天堂| 久久婷五月综合| 国产精品国码视频| 日韩电影大全在线观看| 久久久久毛片免费观看| 4p变态网欧美系列| 高清全集视频免费在线| 精品香蕉在线观看视频一| 中文字幕人妻一区二区在线视频| 伊人婷婷欧美激情| 中文字幕在线看高清电影| 国产米奇在线777精品观看| 国产精品无码一区二区在线| 久久裸体网站| 久久99精品久久久久久三级| 日韩欧美激情| 日本亚洲欧洲色α| 在线免费观看的av| 亚洲色图日韩av| 国精品人妻无码一区二区三区喝尿| 一本大道久久a久久综合| 欧美色图一区二区| 国产色综合一区| 欧美日韩人妻精品一区在线| 麻豆精品一区二区| 免费无遮挡无码永久视频| 亚洲欧美网站在线观看| 日韩和欧美的一区二区| 成人av综合网| 91免费电影网站| 成人看片网站| 69久久夜色精品国产7777| 黄色免费网站在线观看| 国产亚洲美女久久| 熟妇人妻中文av无码| 日韩欧美亚洲国产另类| 中文字幕日韩经典| 色综合视频在线观看| 久久久精品视频免费| 亚洲免费观看视频| 久久久久人妻一区精品色| 久久亚洲综合色一区二区三区| 国内精品免费视频| 国产美女精品在线| 91福利免费观看| 蜜臀久久99精品久久久画质超高清| 2018国产在线| 亚洲精选久久| 免费网站永久免费观看| 欧美一区二区| 人人妻人人澡人人爽精品欧美一区| 禁断一区二区三区在线| 欧美日韩一区二区视频在线观看| 国产成人高清精品免费5388| 999国内精品视频在线| 国产精品视频一区视频二区| 成人免费福利视频| 国产精品久久久久久久久久辛辛 | www.黄色小说.com| 欧美一区二区三区播放老司机| 一区二区久久精品66国产精品| 在线中文字幕一区| 91麻豆精品在线| 在线免费不卡视频| 97人妻精品视频一区| 欧美综合视频在线观看| 成人a v视频| 在线看不卡av| 在线观看国产小视频| 欧美高清视频不卡网| 国产欧美日韩成人| 日韩一本二本av| 韩国中文字幕hd久久精品| 精品欧美一区二区在线观看| 免费观看毛片网站| 日韩电影免费观看中文字幕| 婷婷婷国产在线视频| 亚洲性69xxxbbb| 91在线网址| 北条麻妃一区二区三区中文字幕| 久久久久久国产精品免费无遮挡| 久久综合电影一区| 成人免费一区二区三区牛牛| 97香蕉久久超级碰碰高清版| 中文字幕在线直播| 国产精品嫩草影院久久久| 亚洲青青一区| 国产欧美一区二区视频| 西瓜成人精品人成网站| 日本在线观看一区| 欧美一区精品| 国产黄色一级网站| 久久国产精品区| 91亚洲一线产区二线产区| 91网上在线视频| 免费成人深夜蜜桃视频| 亚洲一区二区三区四区在线观看 | 久久久999免费视频| 免费成人av资源网| 国产又粗又猛又爽又黄| 91丝袜呻吟高潮美腿白嫩在线观看| 无码少妇一区二区| 一区二区三区日韩欧美| 亚洲影院在线播放| 正在播放一区二区| 五月婷婷六月丁香综合| 中文字幕最新精品| av电影免费在线看| 国产精品一区二区三区毛片淫片| 国产精品成av人在线视午夜片| 国产精品中文| 日本黑人久久| 国产主播精品| 日韩精品你懂的| 9i在线看片成人免费| jizzjizzjizz国产| 午夜av电影一区| 国产精品美女一区| 亚洲欧美激情视频| 免费污视频在线观看| 国产精品一区二区久久| 久草精品视频| 青青草免费在线视频观看| 老鸭窝91久久精品色噜噜导演| 亚洲一二区在线观看| 国产视频一区在线观看| 日本一级一片免费视频| 欧美一区二区三区日韩| 黄色av免费在线看| 午夜精品久久久99热福利| av在线国产精品| 日韩在线三区| 日韩亚洲国产精品| 丰满少妇中文字幕| 中文字幕一区二区在线观看| 亚洲不卡在线视频| 亚洲福利视频网| 欧美家庭影院| 91aaaa| 成人三级视频| 久热免费在线观看| 97精品国产露脸对白| 精品无码m3u8在线观看| 这里只有精品视频在线观看| 国产女人在线视频| 国产成人免费91av在线| 亚洲成人一品| 浮妇高潮喷白浆视频| 成人福利视频网站| 国产一级做a爱免费视频| 91精品国产一区二区三区蜜臀| av在线电影网| 国产精品18久久久久久麻辣| 亚洲专区视频| 免费日韩视频在线观看| 91麻豆免费观看| 中文字幕视频网站| 日韩精品电影网| 国产高清自产拍av在线| 国产一区视频观看| 一区二区激情| 亚洲自拍偷拍一区二区| 狠狠色狠色综合曰曰| 无码国产精品一区二区色情男同 | 国产亚洲一区二区手机在线观看 | 你懂的视频在线免费| 2019中文字幕全在线观看| 欧美丝袜美腿| 奇米精品一区二区三区| 久久嫩草精品久久久久| 无码人妻精品一区二区三区蜜桃91| 亚洲欧美日韩久久久久久| **欧美日韩在线观看| 日韩欧美视频一区二区| 美女网站色91| 日韩精品一区二区三区在线视频| 欧美性受xxxx黑人xyx性爽| 日本中文在线| 成人18视频| 国产麻豆综合| 日本人亚洲人jjzzjjz| 欧美精品三级日韩久久| 天堂va在线| 久久av一区二区三区漫画| 日韩精品一区第一页| 亚洲 欧美 国产 另类| 欧美一级久久久| 97超碰免费在线| 日韩欧美99| 国产精品中文字幕日韩精品| 亚洲国产成人精品激情在线| 亚洲色图15p| 欧美日本三级| 无码人妻精品一区二区三区在线| 久久久国产综合精品女国产盗摄| 这里只有精品999| 欧美另类第一页| 国产91精品对白在线播放| 中文字幕12页| 天天影视色香欲综合网老头| 大胆av不用播放器在线播放 | 欧美视频亚洲图片| 性感美女久久精品| www.亚洲资源| 国产伦精品一区二区三区在线| 久热国产精品| 清纯粉嫩极品夜夜嗨av| 亚洲欧美在线播放| 精品国产亚洲一区二区三区| 欧美在线观看成人| 亚洲欧美电影一区二区| 日中文字幕在线| 91老司机精品视频| 亚洲在线观看| avtt天堂在线| 亚洲视频欧洲视频| 91麻豆精品激情在线观看最新 | 亚洲a级精品| 两女双腿交缠激烈磨豆腐 | 亚洲va欧美va|