偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

細(xì)數(shù)ThreadLocal三大坑,內(nèi)存泄露僅是小兒科

開發(fā) 后端
ThreadLocal寫錯(cuò)難,但是用錯(cuò)就很容易,本文將會(huì)詳細(xì)總結(jié)ThreadLocal容易用錯(cuò)的三個(gè)坑。

 

[[398500]]

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

內(nèi)存泄露

線程池中線程上下文丟失

并行流中線程上下文丟失

內(nèi)存泄露

由于ThreadLocal的key是弱引用,因此如果使用后不調(diào)用remove清理的話會(huì)導(dǎo)致對應(yīng)的value內(nèi)存泄露。 

  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<>();  

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

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

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

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

熟悉Tomcat的同學(xué)知道,Tomcat中的web應(yīng)用由Webapp Classloader這個(gè)類加載器的,并且Webapp Classloader是破壞雙親委派機(jī)制實(shí)現(xiàn)的,即所有的web應(yīng)用先由Webapp classloader加載,這樣的好處就是可以讓同一個(gè)容器中的web應(yīng)用以及依賴隔離。

下面我們看具體的內(nèi)存泄露的例子: 

  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.  }  

需要注意這個(gè)例子中的兩個(gè)非常關(guān)鍵的點(diǎn):

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

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

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

線程池中線程上下文丟失

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

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

提交的任務(wù)定義長這樣: 

  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.             // 執(zhí)行業(yè)務(wù)邏輯  
  11.         } catch(Exception e){  
  12.             //log error  
  13.         } finally {  
  14.             ContextHolder.remove(); // 清理 ThreadLocal 的上下文,避免線程復(fù)用時(shí)context互串  
  15.         }  
  16.         return null;  
  17.     }  

對應(yīng)的線程上下文管理類為: 

  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.     }     

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

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

其中最后一個(gè)參數(shù)控制著當(dāng)線程池滿時(shí),該如何處理提交的任務(wù),內(nèi)置有4種策略 

  1. ThreadPoolExecutor.AbortPolicy //直接拋出異常  
  2. ThreadPoolExecutor.DiscardPolicy //丟棄當(dāng)前任務(wù)  
  3. ThreadPoolExecutor.DiscardOldestPolicy //丟棄工作隊(duì)列頭部的任務(wù)  
  4. ThreadPoolExecutor.CallerRunsPolicy //轉(zhuǎn)串行執(zhí)行 

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

并行流中線程上下文丟失

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

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

這段代碼很容易在線下測試的過程中發(fā)現(xiàn)不能按照預(yù)期工作,因?yàn)椴⑿辛鞯讓拥膶?shí)現(xiàn)也是一個(gè)ForkJoin線程池,既然是線程池,那ContextHolder.get()可能取出來的就是一個(gè)null。我們順著這個(gè)思路把代碼再改一下: 

  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.         // 先校驗(yàn)參數(shù),篇幅限制先省略不寫  
  8.         dataList.parallelStream().forEach(entry -> {  
  9.             try {  
  10.                 ContextHolder.set(session);  
  11.                 // 業(yè)務(wù)處理  
  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.     }  

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

 

責(zé)任編輯:龐桂玉 來源: Java知音
相關(guān)推薦

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

內(nèi)存JVMEntry

2024-10-31 09:24:42

2019-08-07 06:16:28

物聯(lián)網(wǎng)IOT技術(shù)

2021-12-28 00:27:24

運(yùn)營商內(nèi)卷網(wǎng)絡(luò)

2023-06-30 08:10:14

JavaBigDecimal

2023-04-16 19:34:01

2022-10-18 08:38:16

內(nèi)存泄漏線程

2018-04-02 07:32:15

2023-02-17 08:20:24

SQL腳本數(shù)據(jù)庫

2017-03-30 08:42:42

技術(shù)信息安全開源

2021-04-23 20:59:02

ThreadLocal內(nèi)存

2022-05-15 08:13:50

Mysql數(shù)據(jù)庫Mycat

2024-05-30 12:15:04

2021-09-26 09:16:45

RedisGeo 類型數(shù)據(jù)類型

2015-11-20 10:39:31

2018-07-19 16:00:25

2010-11-08 13:39:05

網(wǎng)宿科技B2C企業(yè)網(wǎng)站
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號