Java併發編程總結5——ThreadPoolExecutor

来源:http://www.cnblogs.com/everSeeker/archive/2016/07/01/5632450.html
-Advertisement-
Play Games

一、ThreadPoolExecutor介紹 在jdk1.8中,構造函數有4個。以 ThreadPoolExecutor(int corePoolSize, int maximumPoolSize, long keepAliveTime, TimeUnit unit, BlockingQueue<R ...


一、ThreadPoolExecutor介紹

在jdk1.8中,構造函數有4個。以

ThreadPoolExecutor(int corePoolSize, int maximumPoolSize, long keepAliveTime, TimeUnit unit, BlockingQueue<Runnable> workQueue, ThreadFactory threadFactory, RejectedExecutionHandler handler)為例:

1、corePoolSize: 核心線程池大小

2、maximumPoolSize: 最大線程池大小

3、keepAliveTime: 當線程池中的線程數大於corePoolSize時, 多餘空閑線程等待新任務的最長時間, 超過這個時間後多餘線程終止

4、unit: 時間單位, 比如毫秒, 納秒等

5、workQueue: 阻塞隊列

6、threadFactory: 創建線程工廠, 可以方便的創建線程, 可以自定義; 預設為Executors.DefaultThreadFactory

7、handler: 飽和策略, 預設為AbortPolicy

 

定義一個完整的線程池可以這麼寫:

    private ThreadPoolExecutor defaultThreadPool = new ThreadPoolExecutor(10, 100, 100, TimeUnit.MILLISECONDS, new LinkedBlockingQueue<Runnable>(1000), Executors.defaultThreadFactory(), new ThreadPoolExecutor.DiscardPolicy());

 

線程池的主要處理流程如下:

1、提交任務到線程池,判斷核心線程池(corePoolSize)是否已滿? 沒滿,創建工作線程執行任務;滿了,進入下個流程。

2、線程池判斷工作隊列(workQueue)是否已滿?沒滿,則將新提交的任務存儲在工作隊列里。滿了,則進入下個流程。

3、判斷整個線程池(maximumPoolSize)是否已滿?沒滿,則創建一個新的工作線程來執行任務,滿了,則交給飽和策略(RejectedExecutionHandler)來處理這個任務。

 

二、飽和策略 RejectedExecutionHandler

在ThreadPoolExecutor類中定義了4種RejectedExecutionHandler類型:

1、AbortPolicy: 預設策略,直接拋出RejectedExecutionException異常。

2、CallerRunsPolicy:只用調用者所線上程來運行任務。

3、DiscardOldestPolicy:丟棄隊列里最近的一個任務,並執行當前任務。

4、DiscardPolicy:不處理,丟棄掉。

也可以自定義飽和策略,比如將無法處理的新任務加入日誌等,只需要實現RejectedExecutionHandler介面即可。

 

三、阻塞隊列BlockingQueue

    /**
     * ArrayBlockingQueue: 由數組結構組成的有界阻塞隊列, 隊列遵循FIFO
     */
    private BlockingQueue<String> abq = new ArrayBlockingQueue<String>(10);

    /**
     * LinkedBlockingQueue: 由鏈表結構組成的阻塞隊列, FIFO
     * LinkedBlockingDeque: 由鏈表結構組成的雙向阻塞隊列, 構造方法和LinkedBlockingQueue類似
     * public LinkedBlockingQueue(int capacity)     //有界阻塞隊列,隊列最大值為capacity
     * public LinkedBlockingQueue() { this(Integer.MAX_VALUE); }        //無界阻塞隊列,隊列最大值為Integer.MAX_VALUE
     * 通過Executors.newFixedThreadPool(int nThreads)創建的線程池中採用的是無界LinkedBlockingQueue
     * 對於put和take操作, 內部採用了不同的鎖: putLock, takeLock, 而ArrayBlockingQueue內部只有一把鎖
     */
    private BlockingQueue<Thread> lbq = new LinkedBlockingQueue<Thread>(10);
    private ExecutorService fixedThreadPool = Executors.newFixedThreadPool(10);
    private BlockingQueue<String> lbd = new LinkedBlockingDeque<String>();

    /**
     * PriorityBlockingQueue: 支持優先順序排序的有界阻塞隊列
     * public PriorityBlockingQueue(int initialCapacity, Comparator<? super E> comparator)
     */
    private BlockingQueue<String> pbq = new PriorityBlockingQueue(100, new Comparator<String>() {
        public int compare(String o1, String o2) {
            return o1.compareTo(o2);        //升序排列
        }
    });

    /**
     * DelayQueue: 一個使用優先順序隊列實現的無界阻塞隊列, 支持延時獲取元素, 適用於緩存系統的設計以及定時任務調度。
     * 內部隊列採用PriorityQueue, private final PriorityQueue<E> q = new PriorityQueue<E>();
     * 隊列元素需要實現Delayed介面, class DelayQueue<E extends Delayed> extends AbstractQueue<E>
     */

    /**
     * SynchronousQueue: 不存儲元素的阻塞隊列
     */

    /**
     * LinkedTransferQueue: 由鏈表結構組成的實現了TransferQueue介面的無界阻塞隊列
     * transfer方法: 如果當前消費者正在等待接收元素(take()或poll(long timeout, TimeUnit unit))方法, 生產者傳入的元素可以直接傳遞給消費者, 而不放入隊列
     */
    BlockingQueue<String> ltq = new LinkedTransferQueue<String>();

 

四、通過Executors創建線程池

       /**
         * 創建單個線程, 適用於需要保證順序執行任務的場景
         * return new FinalizableDelegatedExecutorService(new ThreadPoolExecutor(1, 1, 0L, TimeUnit.MILLISECONDS, new LinkedBlockingQueue<Runnable>()));
         * BlockingQueue採用無界LinkedBlockingQueue, 因此maximumPoolSize、keepAliveTime參數無意義
         */
        Executors.newSingleThreadExecutor();
/** * 創建固定線程, 適用於需要限制當前線程數量, 負載比較重的伺服器 * return new ThreadPoolExecutor(nThreads, nThreads, 0L, TimeUnit.MILLISECONDS, new LinkedBlockingQueue<Runnable>()); * 採用無界LinkedBlockingQueue, 因此maximumPoolSize、keepAliveTime參數無意義 */ Executors.newFixedThreadPool(10);
/** * 根據需要創建線程, 適用於執行很多的短期非同步任務的小程式, 或者負載較輕的伺服器 * return new ThreadPoolExecutor(0, Integer.MAX_VALUE, 60L, TimeUnit.SECONDS, new SynchronousQueue<Runnable>()); * 初始線程為0, 採用SynchronousQueue阻塞隊列, 如果生產任務的速度低於消費的速度, 空閑60s的線程會被終止; 如果生產任務的速度持續高於消費速度, 則會不斷創建新線程 */ Executors.newCachedThreadPool();
/** * 在給定的延遲之後運行任務, 或者定期執行任務 * super(corePoolSize, Integer.MAX_VALUE, 0, NANOSECONDS, new DelayedWorkQueue()); * 採用無界DelayQueue隊列, 因此maximumPoolSize、keepAliveTime參數無意義 */ Executors.newScheduledThreadPool(10);

 

 

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 做著項目突然SVN報Previous operation has not finished; run 'cleanup' if it was interrupted,進度又要繼續,煩。百度一下發現很多解決方案,自己馬上嘗試的解決,還有點意思,記錄一下。 1.下載sqlite3.exe 2.找到你項目 ...
  • 1 template模版文件uploadfile.html 特別註意的是,只有當request方法是POST,且發送request的<form>有屬性enctype="multipart/form-data"時,request.FILES中包含文件數據,否則request.FILES為空。 2 視圖 ...
  • 題目:輸入n個整數,找出其中最小的K個數。例如輸入4,5,1,6,2,7,3,8這8個數字,則最小的4個數字是1,2,3,4,。 思路1:我們通過快排找到第k個數,然後比他的小的都在左邊,比他大的都在右邊。 思路2:應對大數據的情況,首先取數組中前k個數字建立大根堆,建立堆之後,從第k+1個元素開始 ...
  • activemq(消息列隊) 主要用於分散式應用,支持消息隊列,消息發佈/訂閱,主要用於非同步和伺服器交換數據 ...
  • 1. ptr->name,等同於列印(*p).name。 2. *ptr->name,因為->的優先順序高於*,所以相當於: *(ptr->name)。即指針首地址的那個字元。 3. *ptr->name++,由於*和++的優先順序相同,而且結合性是由右至左,所以相當於: *((ptr->name)++ ...
  • 編寫一個簡易的Servlet計算器,暫時僅能實現 + - * / % 五種運算 jsp界面: 1 <%@ page language="java" contentType="text/html; charset=UTF-8" 2 pageEncoding="UTF-8"%> 3 <!DOCTYPE ...
  • 這個類表示在單獨的控制線程中運行的活動。有兩種方法可以指定這種活動,給構造函數傳遞迴調對象,或者在子類中重寫run() 方法。其他方法(除了構造函數)都不應在子類中被重寫。換句話說,在子類中只有__init__()和run()方法被重寫。 一旦線程對象被創建,它的活動需要通過調用線程的start() ...
  • PHP結構語句 順序結構 順序結構就像一條直線,按著順序一直往下執行。我們編寫的代碼預設都是按照順序結構執行的。 條件結構之if…else… 條件結構就像一個岔路口,可以向左走,也可以向右走。比如上洗手間,我們知道我們的性 別,這時候我們需要根據洗手間提供的條件,左邊男洗手間,右邊女洗手間,或者正好 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...