精品免费在线观看-精品欧美-精品欧美成人bd高清在线观看-精品欧美高清不卡在线-精品欧美日韩一区二区

17站長(zhǎng)網(wǎng)

Tomcat修正JDK原生線程池bug的實(shí)現(xiàn)原理

2022-12-4 12:21| 查看: 2318 |來源: 互聯(lián)網(wǎng)

大家好,現(xiàn)在小編來為大家解答以上問題。Tomcat修正JDK原生線程池bug的實(shí)現(xiàn)原理相信很多人還不知道,現(xiàn)在讓我們一起來看看吧!為提高處理能力和并發(fā)度,Web容器一般會(huì)把處理請(qǐng)求的任務(wù)放到線程池,而JDK的原生線程池

大家好,現(xiàn)在小編來為大家解答以上問題。Tomcat修正JDK原生線程池bug的實(shí)現(xiàn)原理相信很多人還不知道,現(xiàn)在讓我們一起來看看吧!

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

Tomcat 線程池原理

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

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

限制隊(duì)列長(zhǎng)度

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

// 定制的任務(wù)隊(duì)列 taskqueue = new TaskQueue(maxQueueSize); // 定制的線程工廠 TaskThreadFactory tf = new TaskThreadFactory(namePrefix, daemon, getThreadPriority() ); // 定制線程池 executor = new ThreadPoolExecutor(getMinSpareThreads(), getMaxThreads(), maxIdleTime, TimeUnit.MILLISECONDS, taskqueue, 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)的呢?

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

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

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

submittedCount.incrementAndGet();

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

submittedCount.decrementAndGet();

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

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

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

public class TaskQueue extends LinkedBlockingQueue { public TaskQueue(int capacity) { super(capacity); } ... }

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ī)? public class TaskQueue extends LinkedBlockingQueue { ... @Override // 線程池調(diào)用任務(wù)隊(duì)列的方法時(shí),當(dāng)前線程數(shù) > core線程數(shù) public boolean offer(Runnable o) { // 若線程數(shù)已達(dá)max,則不能創(chuàng)建新線程,只能放入任務(wù)隊(duì)列 if (parent.getPoolSize() == parent.getMaximumPoolSize()) return super.offer(o); // 至此,表明 max線程數(shù) > 當(dāng)前線程數(shù) > core線程數(shù) // 說明可創(chuàng)建新線程: // 1. 若已提交任務(wù)數(shù) < 當(dāng)前線程數(shù) // 表明還有空閑線程,無需創(chuàng)建新線程 if (parent.getSubmittedCount()<=(parent.getPoolSize())) return super.offer(o); // 2. 若已提交任務(wù)數(shù) > 當(dāng)前線程數(shù) // 線程不夠用了,返回false去創(chuàng)建新線程 if (parent.getPoolSize()

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

以上Tomcat修正JDK原生線程池bug的實(shí)現(xiàn)原理就是小編為大家收集整理的全部?jī)?nèi)容了,希望對(duì)大家有所幫助。如果您喜歡這篇文章,可以收藏或分享給您的小伙伴們吧!歡迎持續(xù)關(guān)注我們的后續(xù)更新。

本文最后更新于 2022-12-4 12:21,某些文章具有時(shí)效性,若有錯(cuò)誤或已失效,請(qǐng)?jiān)诰W(wǎng)站留言或聯(lián)系站長(zhǎng):[email protected]
·END·
站長(zhǎng)網(wǎng)微信號(hào):w17tui,關(guān)注站長(zhǎng)、創(chuàng)業(yè)、關(guān)注互聯(lián)網(wǎng)人 - 互聯(lián)網(wǎng)創(chuàng)業(yè)者營(yíng)銷服務(wù)中心

免責(zé)聲明:本站部分文章和圖片均來自用戶投稿和網(wǎng)絡(luò)收集,旨在傳播知識(shí),文章和圖片版權(quán)歸原作者及原出處所有,僅供學(xué)習(xí)與參考,請(qǐng)勿用于商業(yè)用途,如果損害了您的權(quán)利,請(qǐng)聯(lián)系我們及時(shí)修正或刪除。謝謝!

17站長(zhǎng)網(wǎng)微信二維碼

始終以前瞻性的眼光聚焦站長(zhǎng)、創(chuàng)業(yè)、互聯(lián)網(wǎng)等領(lǐng)域,為您提供最新最全的互聯(lián)網(wǎng)資訊,幫助站長(zhǎng)轉(zhuǎn)型升級(jí),為互聯(lián)網(wǎng)創(chuàng)業(yè)者提供更加優(yōu)質(zhì)的創(chuàng)業(yè)信息和品牌營(yíng)銷服務(wù),與站長(zhǎng)一起進(jìn)步!讓互聯(lián)網(wǎng)創(chuàng)業(yè)者不再孤獨(dú)!

掃一掃,關(guān)注站長(zhǎng)網(wǎng)微信

大家都在看

熱門排行

    最近更新

      返回頂部
      主站蜘蛛池模板: 中国一级黄色 | 亚洲在线视频免费 | 桃花福利视频在线观看 | 日本韩国欧美在线观看 | 久久免费国产精品一区二区 | 伊人久久综合视频 | 久久15| 国产丝袜视频在线观看 | 日韩视频一区二区 | 免费一级特黄 欧美大片 | 亚洲一区亚洲二区 | 51久久夜色精品国产 | 亚洲欧美日韩在线精品一区二区 | 欧美成人高清乱码 | 日韩在线观看精品 | 色黄啪啪18周岁以下禁止观看 | 大学生久久香蕉国产线看观看 | 久热精品在线视频 | 亚欧毛片 | 91精品国产高清91久久久久久 | 全黄毛片 | 亚洲黄a| 福利一区二区三区视频午夜观看 | 久久国产精品最新一区 | 日韩国产欧美一区二区三区在线 | 午夜网站在线播放 | 色婷婷狠狠 | 欧美三级成人观看 | 丁香六月激情婷婷 | 黄色自拍 | 欧美三级毛片 | 国产黄色在线播放 | 精品国产成人三级在线观看 | 中文字幕a∨在线乱码免费看 | 毛色毛片免费观看 | 国产精品第一页爽爽影院 | 国产成人激烈叫床声视频对白 | 香蕉视频免费在线观看 | yjizz视频 | 亚洲欧美黄色片 | 一级网站在线观看 |