{ return "hello word" ;}); 還有Tomcat中的線程池 org.apache.tomcat.util.threads.TaskQueue org.apache.tomcat.util.threads.ThreadPoolExecutor 線程池維護多個線程,等待監督管理者分配可并發執行的任務。這種做法,一方面避免了處理任務時創建銷毀線程開銷的代價,另一方面避免了線程數量膨脹導致的過分調度問題,保證了對" />

衡阳派盒市场营销有限公司

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

線程池的兩個思考

科技綠洲 ? 來源:Java技術指北 ? 作者:Java技術指北 ? 2023-09-30 11:21 ? 次閱讀

今天還是說一下線程池的兩個思考。

池子

我們常用的線程池,

  1. JDK的ThreadPoolExecutor.
  2. CompletableFutures 默認使用了ForkJoinPool.commonPool()。
CompletableFuture.supplyAsync(()- >{ return "hello word";});
  1. 還有Tomcat中的線程池

org.apache.tomcat.util.threads.TaskQueue

org.apache.tomcat.util.threads.ThreadPoolExecutor

線程池維護多個線程,等待監督管理者分配可并發執行的任務。這種做法,一方面避免了處理任務時創建銷毀線程開銷的代價,另一方面避免了線程數量膨脹導致的過分調度問題,保證了對內核的充分利用。

JDK 線程池

public ThreadPoolExecutor(
    int corePoolSize, //核心線程數
    int maximumPoolSize,//最大線程數
    long keepAliveTime, //大于核心線程數量的線程存活時間,如果沒有新任務就會關閉
    TimeUnit unit, // 時間單位
    BlockingQueue< Runnable > workQueue, //線程等待隊列
    ThreadFactory threadFactory,//創建線程的工廠
    RejectedExecutionHandler handler//拒絕策略
) {

JDK線程池執行任務:

  1. 提交任務給線程池后,線程池會檢查線程池中正在運行的線程數量,如果線程數量小于核心線程,則創建一個新的線程來處理任務。
  2. 如果線程池中線程數量達到和corePoolSize的大小,則將線程放入等待隊列BlockingQueue中。
  3. 如果提交任務時連等待隊列都已經滿了的話,線程池會繼續創建新的線程來處理任務,直到線程池數量達到maximumPoolSize。
  4. 如果線程數量達到了最大容量,則會執行拒絕策略。

這里需要注意直接使用LinkedBlockingQueue阻塞隊列作為線程池會存在一個問題,當workcount > corePool時優先進入隊列排隊, 當請求并發過多時會導致請求緩慢,隊列太長可能會出現內存溢出(先排隊再漲線程池)

Tomcat線程池

下面時Tomcat線程池的構造方法

public ThreadPoolExecutor(int corePoolSize, int maximumPoolSize, long keepAliveTime, TimeUnit unit, BlockingQueue< Runnable > workQueue, ThreadFactory threadFactory, RejectedExecutionHandler handler) {
    this.ctl = new AtomicInteger(ctlOf(-536870912, 0));
    this.mainLock = new ReentrantLock();
    this.workers = new HashSet();
    this.termination = this.mainLock.newCondition();
    this.submittedCount = new AtomicInteger(0);
    this.lastContextStoppedTime = new AtomicLong(0L);
    this.lastTimeThreadKilledItself = new AtomicLong(0L);
    this.threadRenewalDelay = 1000L;
    if (corePoolSize >= 0 && maximumPoolSize > 0 && maximumPoolSize >= corePoolSize && keepAliveTime >= 0L) {
        if (workQueue != null && threadFactory != null && handler != null) {
            this.corePoolSize = corePoolSize;
            this.maximumPoolSize = maximumPoolSize;
            this.workQueue = workQueue;
            this.keepAliveTime = unit.toNanos(keepAliveTime);
            this.threadFactory = threadFactory;
            this.handler = handler;
            this.prestartAllCoreThreads();
        } else {
            throw new NullPointerException();
        }
    } else {
        throw new IllegalArgumentException();
    }
}

Tomcat主要針對web接口請求,不能因為LinkedBlockingQueue的排隊導致接口出現大量延遲和緩慢, 從而使用了tomcat的TaskQueue,TaskQueue繼承了JDK的LinkedBlockingQueue 并擴展了JDK線程池的功能。

主要有一下幾點優化:

  • Tomcat的ThreadPoolExecutor使用的TaskQueue,是無界的LinkedBlockingQueue,但是通過taskQueue的offer方法覆蓋了LinkedBlockingQueue的offer方法,修改了線程池增長規則,使得線程池能在任務較多的情況下增長線程池數量。(先漲線程池再排隊。)
  • Tomcat的ThreadPoolExecutor改寫了execute方法,當任務被reject時,捕獲異常,會強制入隊
public void execute(Runnable command, long timeout, TimeUnit unit) {
    this.submittedCount.incrementAndGet();

    try {
        this.executeInternal(command);
    } catch (RejectedExecutionException var9) {
        if (!(this.getQueue() instanceof TaskQueue)) {
            this.submittedCount.decrementAndGet();
            throw var9;
        }

        TaskQueue queue = (TaskQueue)this.getQueue();

        try {
            if (!queue.force(command, timeout, unit)) {
                this.submittedCount.decrementAndGet();
                throw new RejectedExecutionException(sm.getString("threadPoolExecutor.queueFull"));
            }
        } catch (InterruptedException var8) {
            this.submittedCount.decrementAndGet();
            throw new RejectedExecutionException(var8);
        }
    }

}

那個線程池適合

我們看看AI如何回復

圖片

了不起認為大多數情況下使用JDK的線程池就夠用了,如果覺得線程數據處理不過來,需要多一點線程直接增加核心線程數量設置就可以了。針對資源比較緊張,對線程使用代價比較高時可以考慮。

tomcat對線程池做過優化,也必然是有一定的考量,對于線程資源的使用頻率比較高的情況下可以使用。

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 內核
    +關注

    關注

    3

    文章

    1382

    瀏覽量

    40422
  • 接口
    +關注

    關注

    33

    文章

    8691

    瀏覽量

    151913
  • 線程池
    +關注

    關注

    0

    文章

    57

    瀏覽量

    6893
  • JDK
    JDK
    +關注

    關注

    0

    文章

    82

    瀏覽量

    16636
  • tomcat
    +關注

    關注

    0

    文章

    30

    瀏覽量

    4871
收藏 人收藏

    評論

    相關推薦

    Java中的線程包括哪些

    java.util.concurrent 包來實現的,最主要的就是 ThreadPoolExecutor 類。 Executor: 代表線程的接口,有一 execute() 方法,給一
    的頭像 發表于 10-11 15:33 ?858次閱讀
    Java中的<b class='flag-5'>線程</b><b class='flag-5'>池</b>包括哪些

    線程創建的種方法

    1. 使用內置模塊在使用多線程處理任務時也不是線程越多越好,由于在切換線程的時候,需要切換上下文環境,依然會造成cpu的大量開銷。為解決這個問題,線程
    發表于 03-16 16:15

    兩個線程和互斥鎖如何形成死循環?

    兩個線程兩個互斥鎖如何形成死鎖?程序流程圖如下: 程序流程圖 如上圖所示: t0時刻,主線程創建子線程,并初始化互斥鎖mutex1、mut
    的頭像 發表于 01-02 16:47 ?1516次閱讀
    <b class='flag-5'>兩個</b><b class='flag-5'>線程</b>和互斥鎖如何形成死循環?

    兩個線程兩個互斥鎖如何形成死鎖

    兩個線程兩個互斥鎖如何形成死鎖? 程序流程圖如下: 程序流程圖 如上圖所示: t0時刻,主線程創建子線程,并初始化互斥鎖mutex1、mu
    的頭像 發表于 12-28 09:24 ?2278次閱讀
    <b class='flag-5'>兩個</b><b class='flag-5'>線程</b>,<b class='flag-5'>兩個</b>互斥鎖如何形成死鎖

    如何正確關閉線程

    前言本章分為兩個議題 如何正確關閉線程 shutdown 和 shutdownNow 的區別 項目環境jdk 1.8 github 地址:https://github.com
    的頭像 發表于 09-29 14:41 ?9999次閱讀

    在Python中用于終止線程兩個選項

    ? 我經常被問到如何殺死一后臺線程,這個問題的答案讓很多人不開心: 線程是殺不死的。在本文中,我將向您展示? Python ?中用于終止線程兩個
    的頭像 發表于 11-17 10:02 ?4729次閱讀

    python創建線程種方法

    在使用多線程處理任務時也不是線程越多越好,由于在切換線程的時候,需要切換上下文環境,依然會造成cpu的大量開銷。為解決這個問題,線程的概念
    的頭像 發表于 03-16 16:15 ?6016次閱讀

    線程線程

    線程通常用于服務器應用程序。 每個傳入請求都將分配給線程池中的一線程,因此可以異步處理請求,而不會占用主
    的頭像 發表于 02-28 09:53 ?832次閱讀
    多<b class='flag-5'>線程</b>之<b class='flag-5'>線程</b><b class='flag-5'>池</b>

    如何用C++實現一線程呢?

    C++線程是一種多線程管理模型,把線程分成任務執行和線程調度部分。
    發表于 06-08 14:53 ?1836次閱讀
    如何用C++實現一<b class='flag-5'>個</b><b class='flag-5'>線程</b><b class='flag-5'>池</b>呢?

    細數線程的10

    JDK開發者提供了線程的實現類,我們基于Executors組件,就可以快速創建一線程
    的頭像 發表于 06-16 10:11 ?765次閱讀
    細數<b class='flag-5'>線程</b><b class='flag-5'>池</b>的10<b class='flag-5'>個</b>坑

    Spring 的線程應用

    我們在日常開發中,經常跟多線程打交道,Spring 為我們提供了一線程方便我們開發,它就是 ThreadPoolTaskExecutor ,接下來我們就來聊聊 Spring 的
    的頭像 發表于 10-13 10:47 ?650次閱讀
    Spring 的<b class='flag-5'>線程</b><b class='flag-5'>池</b>應用

    線程基本概念與原理

    、17、20等的新特性,簡化了多線程編程的實現。 提高性能與資源利用率 線程主要解決兩個問題:線程創建與銷毀的開銷以及
    的頭像 發表于 11-10 10:24 ?578次閱讀

    線程的基本概念

    ? 呃呃,我這么問就很奇怪,因為線程是什么我都沒說,怎么會知道為什么會有線程呢?所以我打算帶大家去思考
    的頭像 發表于 11-10 16:37 ?560次閱讀
    <b class='flag-5'>線程</b><b class='flag-5'>池</b>的基本概念

    線程的創建方式有幾種

    線程是一種用于管理和調度線程的技術,能夠有效地提高系統的性能和資源利用率。它通過預先創建一組線程并維護一工作隊列,將任務提交給
    的頭像 發表于 12-04 16:52 ?933次閱讀

    什么是動態線程?動態線程的簡單實現思路

    因此,動態可監控線程一種針對以上痛點開發的線程管理工具。主要可實現功能有:提供對 Spring 應用內線程
    的頭像 發表于 02-28 10:42 ?721次閱讀
    大发体育| 郸城县| 百家乐官网稳赚的方法| 南京百家乐官网赌博现场被| 百家乐补牌规制| 凤凰百家乐的玩法技巧和规则 | 南京百家乐官网电| 极速百家乐真人视讯| 路劲太阳城业主论坛| 巩留县| 百家乐官网现场新全讯网| 百家乐官网77scs| 戒掉百家乐的玩法技巧和规则 | 百家乐官网凯时娱乐网| 百家乐棋牌交友| 大发888优惠代码 官网| 百家乐官网的规则博彩正网| 百家乐有多少局| 百家乐皇室百家乐的玩法技巧和规则| 贵族娱乐城| 百家乐官网荷| 大发888娱乐城dknmwd| 新澳博百家乐的玩法技巧和规则 | 百家乐官网闲9点| 百家乐注册彩金| 迪威网上娱乐| 英皇百家乐官网的玩法技巧和规则 | 太阳城百家乐的分数| 大连娱网棋牌大厅| 澳门百家乐官网官网站| 中国百家乐官网技巧软件| 新澳博百家乐的玩法技巧和规则| 凯旋门百家乐官网现金网| 百家乐博彩开户博彩通| 在线玩轮盘| 做生意门面朝向风水| 如何胜百家乐的玩法技巧和规则| 百家乐官网单跳投注法| 圣安娜百家乐代理| 霍山县| 百家乐暗红色桌布|