一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務(wù)器之家:專注于服務(wù)器技術(shù)及軟件下載分享
分類導(dǎo)航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術(shù)|正則表達(dá)式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務(wù)器之家 - 編程語言 - Java教程 - Tomcat如何修正JDK原生線程池Bug?

Tomcat如何修正JDK原生線程池Bug?

2021-08-20 23:22JavaEdge Java教程

為提高處理能力和并發(fā)度,Web容器一般會(huì)把處理請(qǐng)求的任務(wù)放到線程池,而JDK的原生線程池先天適合CPU密集型任務(wù),并不適合我們通常的 I/O 密集任務(wù)處理,于是Tomcat改造之。

Tomcat如何修正JDK原生線程池Bug?

為提高處理能力和并發(fā)度,Web容器一般會(huì)把處理請(qǐng)求的任務(wù)放到線程池,而JDK的原生線程池先天適合CPU密集型任務(wù),并不適合我們通常的 I/O 密集任務(wù)處理,于是Tomcat改造之。

Tomcat 線程池原理

其實(shí)ThreadPoolExecutor的參數(shù)主要有如下關(guān)鍵點(diǎn):

  • 限制線程個(gè)數(shù) 

Tomcat如何修正JDK原生線程池Bug?

  • 限制隊(duì)列長度

Tomcat如何修正JDK原生線程池Bug?

而Tomcat對(duì)這倆資源都需要限制,否則高并發(fā)下CPU、內(nèi)存都有被耗盡可能。因此Tomcat的線程池傳參:

  1. // 定制的任務(wù)隊(duì)列 
  2. taskqueue = new TaskQueue(maxQueueSize); 
  3.  
  4. // 定制的線程工廠 
  5. TaskThreadFactory tf = new TaskThreadFactory(namePrefix, 
  6.                                daemon, 
  7.                                getThreadPriority() 
  8. ); 
  9.  
  10. // 定制線程池 
  11. executor = new ThreadPoolExecutor(getMinSpareThreads(), 
  12.                   getMaxThreads(), 
  13.                      maxIdleTime,  
  14.                      TimeUnit.MILLISECONDS, 
  15.                      taskqueue, 
  16.                      tf); 

Tomcat對(duì)線程數(shù)也有限制,設(shè)置:

  • 核心線程數(shù)(minSpareThreads)
  • 最大線程池?cái)?shù)(maxThreads)

Tomcat線程池還有自己的特色任務(wù)處理流程,通過重寫execute方法實(shí)現(xiàn)了自己的特色任務(wù)處理邏輯:

  1. 前corePoolSize個(gè)任務(wù)時(shí),來一個(gè)任務(wù)就創(chuàng)建一個(gè)新線程
  2. 再有任務(wù),就把任務(wù)放入任務(wù)隊(duì)列,讓所有線程去搶。若隊(duì)列滿,就創(chuàng)建臨時(shí)線程
  3. 總線程數(shù)達(dá)到maximumPoolSize,則繼續(xù)嘗試把任務(wù)放入任務(wù)隊(duì)列
  4. 若緩沖隊(duì)列也滿了,插入失敗,執(zhí)行拒絕策略

和 JDK 線程池的區(qū)別就在step3,Tomcat在線程總數(shù)達(dá)到最大數(shù)時(shí),不是立即執(zhí)行拒絕策略,而是再嘗試向任務(wù)隊(duì)列添加任務(wù),添加失敗后再執(zhí)行拒絕策略。

具體又是如何實(shí)現(xiàn)的呢?

Tomcat如何修正JDK原生線程池Bug?

  1. public void execute(Runnable command, long timeout, TimeUnit unit) { 
  2.     submittedCount.incrementAndGet(); 
  3.     try { 
  4.         // 調(diào)用JDK原生線程池的execute執(zhí)行任務(wù) 
  5.         super.execute(command); 
  6.     } catch (RejectedExecutionException rx) { 
  7.        // 總線程數(shù)達(dá)到maximumPoolSize后,JDK原生線程池會(huì)執(zhí)行默認(rèn)拒絕策略 
  8.         if (super.getQueue() instanceof TaskQueue) { 
  9.             final TaskQueue queue = (TaskQueue)super.getQueue(); 
  10.             try { 
  11.                 // 繼續(xù)嘗試把任務(wù)放入任務(wù)隊(duì)列 
  12.                 if (!queue.force(command, timeout, unit)) { 
  13.                     submittedCount.decrementAndGet(); 
  14.                     // 若緩沖隊(duì)列還是滿了,插入失敗,執(zhí)行拒絕策略。 
  15.                     throw new RejectedExecutionException("..."); 
  16.                 } 
  17.             }  
  18.         } 
  19.     } 

定制任務(wù)隊(duì)列

Tomcat線程池的execute方法第一行:

  1. submittedCount.incrementAndGet(); 

任務(wù)執(zhí)行失敗,拋異常時(shí),將該計(jì)數(shù)器減一:

  1. submittedCount.decrementAndGet(); 

Tomcat線程池使用 submittedCount 變量維護(hù)已提交到線程池,但未執(zhí)行完的任務(wù)數(shù)量。

為何要維護(hù)這樣一個(gè)變量呢?

Tomcat的任務(wù)隊(duì)列TaskQueue擴(kuò)展了JDK的LinkedBlockingQueue,Tomcat給了它一個(gè)capacity,傳給父類LinkedBlockingQueue的構(gòu)造器。

  1. public class TaskQueue extends LinkedBlockingQueue<Runnable> { 
  2.  
  3.   public TaskQueue(int capacity) { 
  4.       super(capacity); 
  5.   } 
  6.   ... 

capacity參數(shù)通過Tomcat的 maxQueueSize 參數(shù)設(shè)置,但maxQueueSize默認(rèn)值為Integer.MAX_VALUE:這樣,當(dāng)前線程數(shù)達(dá)到核心線程數(shù)后,再來的任務(wù),線程池會(huì)把任務(wù)添加到任務(wù)隊(duì)列,并且總會(huì)成功,就永遠(yuǎn)無機(jī)會(huì)創(chuàng)建新線程了。

為此,TaskQueue重寫了LinkedBlockingQueue#offer,在合適時(shí)機(jī)返回false,表示任務(wù)添加失敗,線程池此時(shí)會(huì)創(chuàng)建新的線程。

什么叫合適時(shí)機(jī)?

  1. public class TaskQueue extends LinkedBlockingQueue<Runnable> { 
  2.  
  3.   ... 
  4.    @Override 
  5.   // 線程池調(diào)用任務(wù)隊(duì)列的方法時(shí),當(dāng)前線程數(shù) > core線程數(shù) 
  6.   public boolean offer(Runnable o) { 
  7.  
  8.       // 若線程數(shù)已達(dá)max,則不能創(chuàng)建新線程,只能放入任務(wù)隊(duì)列 
  9.       if (parent.getPoolSize() == parent.getMaximumPoolSize())  
  10.           return super.offer(o); 
  11.            
  12.       // 至此,表明 max線程數(shù) > 當(dāng)前線程數(shù) > core線程數(shù) 
  13.       // 說明可創(chuàng)建新線程: 
  14.        
  15.       // 1. 若已提交任務(wù)數(shù) < 當(dāng)前線程數(shù) 
  16.       //    表明還有空閑線程,無需創(chuàng)建新線程 
  17.       if (parent.getSubmittedCount()<=(parent.getPoolSize()))  
  18.           return super.offer(o); 
  19.            
  20.       // 2. 若已提交任務(wù)數(shù) > 當(dāng)前線程數(shù) 
  21.       //    線程不夠用了,返回false去創(chuàng)建新線程 
  22.       if (parent.getPoolSize()<parent.getMaximumPoolSize())  
  23.           return false
  24.            
  25.       // 默認(rèn)情況下總是把任務(wù)放入任務(wù)隊(duì)列 
  26.       return super.offer(o); 
  27.   } 
  28.    

所以Tomcat維護(hù) 已提交任務(wù)數(shù) 是為了在任務(wù)隊(duì)列長度無限時(shí),讓線程池還能有機(jī)會(huì)創(chuàng)建新線程。

原文鏈接:https://mp.weixin.qq.com/s/7MOOhnI5qsoi8uAihLXy7A

延伸 · 閱讀

精彩推薦
  • Java教程淺談java 中equals和==的區(qū)別

    淺談java 中equals和==的區(qū)別

    這篇文章主要介紹了java 中equals和==的區(qū)別,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小...

    獨(dú)特潤許多人5982021-07-21
  • Java教程mybatis批量新增、刪除、查詢和修改方式

    mybatis批量新增、刪除、查詢和修改方式

    這篇文章主要介紹了mybatis批量新增、刪除、查詢和修改方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教...

    xuforeverlove7492022-01-24
  • Java教程Java開發(fā)常見異常及解決辦法詳解

    Java開發(fā)常見異常及解決辦法詳解

    這篇文章主要介紹了java程序常見異常及處理匯總,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考...

    cutercorley12252021-12-18
  • Java教程Spring Cloud Gateway 如何修改HTTP響應(yīng)信息

    Spring Cloud Gateway 如何修改HTTP響應(yīng)信息

    這篇文章主要介紹了Spring Cloud Gateway 修改HTTP響應(yīng)信息的方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教...

    帷幄庸者13712021-10-13
  • Java教程Spring 6.0 將停止支持 Freemarker 和 JSP

    Spring 6.0 將停止支持 Freemarker 和 JSP

    Spring Framework 6.0 第一個(gè)里程碑版本已經(jīng)發(fā)布,目前已經(jīng)可以從Spring Repo獲取。這里有一些新變更我們可以提前了解一下。...

    碼農(nóng)小胖哥12642021-12-31
  • Java教程淺談sql_@SelectProvider及使用注意說明

    淺談sql_@SelectProvider及使用注意說明

    這篇文章主要介紹了sql_@SelectProvider及使用注意說明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教...

    icecoola_6892021-11-04
  • Java教程二進(jìn)制中1的個(gè)數(shù)

    二進(jìn)制中1的個(gè)數(shù)

    這篇文章介紹了二進(jìn)制中1的個(gè)數(shù),有需要的朋友可以參考一下 ...

    java之家2662019-10-15
  • Java教程mybatis調(diào)用存儲(chǔ)過程的實(shí)例代碼

    mybatis調(diào)用存儲(chǔ)過程的實(shí)例代碼

    這篇文章主要介紹了mybatis調(diào)用存儲(chǔ)過程的實(shí)例,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下...

    動(dòng)力節(jié)點(diǎn)11732021-01-25
主站蜘蛛池模板: 亚洲国产AV一区二区三区四区 | 天选之王漫画顾长歌免费阅读 | 国产全部理论片线观看 | 精品综合久久久久久8888 | 范冰冰特黄xx大片 | 亚洲日本中文字幕天堂网 | 黑人巨大爆粗亚裔女人 | 日本不卡在线一区二区三区视频 | 欧美日韩亚洲第一区在线 | 波多野结衣中文字幕乱七八糟 | k逼| 四虎影视永久在线 | 免费在线看a | 国产在线观看91精品一区 | 奇米7777第四色 | 日韩毛片在线视频 | 国产成人在线视频 | 日本人在线看片 | 午夜影院一区二区三区 | 精品一区二区三区波多野结衣 | 国产精品久久久久毛片 | 久久99精品涩AV毛片观看 | 国产一区二区三区久久精品 | 91嫩草私人成人亚洲影院 | 国产品精人成福利视频 | 国产免费好大好硬视频 | 免费观看欧美一级高清 | 色婷婷影院在线视频免费播放 | 免费看美女被靠到爽的视频 | 免费国产成人 | 欧美视频在线一区二区三区 | 国产成人在线播放视频 | 四虎精品免费视频 | 成人久久18免费网站 | 饭冈加奈子黑人解禁在线播放 | 亚洲成aⅴ人片在线 | 午夜办公室 | 韩国最新理论三级在线观看 | 四虎成人国产精品视频 | a v在线男人的天堂观看免费 | 波多野结衣之高校教师 |