搜尋
首頁Javajava教程Java自備的四種執行緒池是什麼?

java預先定義的哪四種執行緒池?

  • newSingleThreadExexcutor:單執行緒數的執行緒池(核心執行緒數=最大執行緒數=1)

  • newFixedThreadPool:固定執行緒數的執行緒池(核心執行緒數=最大執行緒數=自訂)

  • newCacheThreadPool:可快取的執行緒池(核心執行緒數=0,最大執行緒數=Integer.MAX_VALUE)

  • newScheduledThreadPool:支援定時或週期任務的執行緒池(核心執行緒數=自訂,最大執行緒數=Integer.MAX_VALUE)

#四個執行緒數池有什麼差別?

上面四種執行緒池類別都繼承ThreadPoolExecutor,在建立時都是直接回傳new ThreadPoolExecutor(參數),它們的差別是定義的ThreadPoolExecutor(參數)中參數不同,而ThreadPoolExecutor又繼承ExecutorService介面類別

  • newFixedThreadPool

定義:
xecutorService executorService=Executors.newFixedThreadPool(2);

Java自備的四種執行緒池是什麼?

##缺點:使用了LinkBlockQueue的鍊錶型阻塞隊列,當任務的堆積速度大於處理速度時,容易堆積任務而導致OOM記憶體溢出

  • newSingleThreadExecutor

定義:ExecutorService executorService =Executors.newSingleThreadExecutor();

Java自備的四種執行緒池是什麼?

##上面程式碼神似似new FixedThreadPoop(1),但又有區別,因為外面多了一層FinalizableDelegatedExecutorService,其作用:

Java自備的四種執行緒池是什麼?##可知,fixedExecutorService的本質是ThreadPoolExecutor,所以fixedExecutorService可以強轉成ThreadPoolExecutor,但singleExecutorService與ThreadPoolExecutor沒有任何關係,所以強轉失敗,故newSingleThreadExecutor()建立後,無法再修改其執行緒池參數,真正做到single單一執行緒。

缺點:使用了LinkBlockQueue的鍊錶型阻塞隊列,當任務的堆積速度大於處理速度時,容易堆積任務而導致OOM記憶體溢出

newCacheThreadPool

#定義:ExecutorService executorService=Executors.newCacheThreadPool();

##缺點:SynchronousQueue是BlockingQueue的實現,它也是一個佇列,因為最大執行緒數為Integer.MAX_VALUE,所有當執行緒過多時容易OOM記憶體溢出Java自備的四種執行緒池是什麼?

ScheduledThreadPool

定義:ExecutorService executorService=Executors.newScheduledThreadPool(2);

##
源码:
public static ScheduledExecutorService newScheduledThreadPool(int corePoolSize) {
        //ScheduledThreadPoolExecutor继承ThreadPoolExecutor
        return new ScheduledThreadPoolExecutor(corePoolSize);
    }
    
public ScheduledThreadPoolExecutor(int corePoolSize) {
    //ScheduledThreadPoolExecutor继承ThreadPoolExecutor,故super()会调用ThreadPoolExecutor的构造函数初始化并返回一个ThreadPoolExecutor,而ThreadPoolExecutor使实现ExecutorService接口的
    //最终ScheduledThreadPoolExecutor也和上面几种线程池一样返回的是ExecutorService接口的实现类ThreadPoolExecutor
    super(corePoolSize, Integer.MAX_VALUE, 0, NANOSECONDS,
          new DelayedWorkQueue());
}
線程池有哪幾個重要參數?

ThreadPoolExecutor建構方法如下:

Java自備的四種執行緒池是什麼?

Java自備的四種執行緒池是什麼?#keepAliveTime是指目前執行緒數位於[核心執行緒數,最大執行緒數] 之間的這些非核心執行緒等待多久空閒時間而沒有活乾時,就退出執行緒池;

  • 等待丟列的大小與最大執行緒數是沒有任何關係的,線程創建優先權=核心線程> 阻塞隊列> 擴容的線程(當前核心線程數小於最大線程數時才能擴容線程)

  • 假如核心線程數5,等待隊列長度為3,最大線程數10:當線程數不斷在增加時,先創建5個核心線程,核心線程數滿了再把線程丟進等待丟列,等待隊列滿了(3個線程),此時會比較最大線程數(只有等待丟列滿了最大線程數才能出場),還可以繼續創建2個線程(5 3 2),若線程數超過了最大線程數,則執行拒絕策略;

  • 假如核心執行緒數5,等待佇列長度為3,最大執行緒數7:當執行緒數不斷在增加時,先建立5個核心線程,核心執行緒數滿了再把線程丟進等待丟列,當等待隊列中有2個線程時達到了最大線程數(5 2=7),但是等待丟列還沒滿所以不用管最大線程數,直到等待丟列滿了(3個阻塞線程),此時會比較最大線程數(只有等待丟列滿了最大線程數才能出場),此時核心等待丟列=5 3=8>7=最大線程數,即已經達到最大執行緒數了,則執行拒絕策略;

  • 如果把等待丟列設定為LinkedBlockingQueue無界丟列,這個丟列是無限大的,就永遠不會走到判斷最大線程數那一步了

如何自定义线程池

可以使用有界队列,自定义线程创建工厂ThreadFactory和拒绝策略handler来自定义线程池

public class ThreadTest {
    public static void main(String[] args) throws InterruptedException, IOException {
        int corePoolSize = 2;
        int maximumPoolSize = 4;
        long keepAliveTime = 10;
        TimeUnit unit = TimeUnit.SECONDS;
        BlockingQueue<Runnable> workQueue = new ArrayBlockingQueue<>(2);
        ThreadFactory threadFactory = new NameTreadFactory();
        RejectedExecutionHandler handler = new MyIgnorePolicy();
       ThreadPoolExecutor executor = new ThreadPoolExecutor(corePoolSize, maximumPoolSize, keepAliveTime, unit,
                workQueue, threadFactory, handler);
        executor.prestartAllCoreThreads(); // 预启动所有核心线程        
        for (int i = 1; i <= 10; i++) {
            MyTask task = new MyTask(String.valueOf(i));
            executor.execute(task);
        }
        System.in.read(); //阻塞主线程
    }
    static class NameTreadFactory implements ThreadFactory {
        private final AtomicInteger mThreadNum = new AtomicInteger(1);
        @Override
        public Thread newThread(Runnable r) {
            Thread t = new Thread(r, "my-thread-" + mThreadNum.getAndIncrement());
            System.out.println(t.getName() + " has been created");
            return t;
        }
    }

    public static class MyIgnorePolicy implements RejectedExecutionHandler {
        @Override
        public void rejectedExecution(Runnable r, ThreadPoolExecutor e) {
            doLog(r, e);
        }
        private void doLog(Runnable r, ThreadPoolExecutor e) {
            // 可做日志记录等
            System.err.println( r.toString() + " rejected");
//          System.out.println("completedTaskCount: " + e.getCompletedTaskCount());
        }
    }

    static class MyTask implements Runnable {
        private String name;
        public MyTask(String name) {
            this.name = name;
        }
        @Override
        public void run() {
            try {
                System.out.println(this.toString() + " is running!");
                Thread.sleep(3000); //让任务执行慢点
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
        public String getName() {
            return name;
        }
        @Override
        public String toString() {
            return "MyTask [name=" + name + "]";
        }
    }
}

运行结果:

Java自備的四種執行緒池是什麼?

其中7-10号线程被拒绝策略拒绝了,1、2号线程执行完后,3、6号线程进入核心线程池执行,此时4、5号线程在任务队列等待执行,3、6线程执行完再通知4、5线程执行

以上是Java自備的四種執行緒池是什麼?的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述
本文轉載於:亿速云。如有侵權,請聯絡admin@php.cn刪除
如何將Maven或Gradle用於高級Java項目管理,構建自動化和依賴性解決方案?如何將Maven或Gradle用於高級Java項目管理,構建自動化和依賴性解決方案?Mar 17, 2025 pm 05:46 PM

本文討論了使用Maven和Gradle進行Java項目管理,構建自動化和依賴性解決方案,以比較其方法和優化策略。

如何使用適當的版本控制和依賴項管理創建和使用自定義Java庫(JAR文件)?如何使用適當的版本控制和依賴項管理創建和使用自定義Java庫(JAR文件)?Mar 17, 2025 pm 05:45 PM

本文使用Maven和Gradle之類的工具討論了具有適當的版本控制和依賴關係管理的自定義Java庫(JAR文件)的創建和使用。

如何使用咖啡因或Guava Cache等庫在Java應用程序中實現多層緩存?如何使用咖啡因或Guava Cache等庫在Java應用程序中實現多層緩存?Mar 17, 2025 pm 05:44 PM

本文討論了使用咖啡因和Guava緩存在Java中實施多層緩存以提高應用程序性能。它涵蓋設置,集成和績效優勢,以及配置和驅逐政策管理最佳PRA

如何將JPA(Java持久性API)用於具有高級功能(例如緩存和懶惰加載)的對象相關映射?如何將JPA(Java持久性API)用於具有高級功能(例如緩存和懶惰加載)的對象相關映射?Mar 17, 2025 pm 05:43 PM

本文討論了使用JPA進行對象相關映射,並具有高級功能,例如緩存和懶惰加載。它涵蓋了設置,實體映射和優化性能的最佳實踐,同時突出潛在的陷阱。[159個字符]

Java的類負載機制如何起作用,包括不同的類載荷及其委託模型?Java的類負載機制如何起作用,包括不同的類載荷及其委託模型?Mar 17, 2025 pm 05:35 PM

Java的類上載涉及使用帶有引導,擴展程序和應用程序類負載器的分層系統加載,鏈接和初始化類。父代授權模型確保首先加載核心類別,從而影響自定義類LOA

See all articles

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

AI Hentai Generator

AI Hentai Generator

免費產生 AI 無盡。

熱門文章

R.E.P.O.能量晶體解釋及其做什麼(黃色晶體)
3 週前By尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.最佳圖形設置
3 週前By尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.如果您聽不到任何人,如何修復音頻
3 週前By尊渡假赌尊渡假赌尊渡假赌
WWE 2K25:如何解鎖Myrise中的所有內容
4 週前By尊渡假赌尊渡假赌尊渡假赌

熱工具

SublimeText3 Linux新版

SublimeText3 Linux新版

SublimeText3 Linux最新版

Dreamweaver Mac版

Dreamweaver Mac版

視覺化網頁開發工具

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

mPDF

mPDF

mPDF是一個PHP庫,可以從UTF-8編碼的HTML產生PDF檔案。原作者Ian Back編寫mPDF以從他的網站上「即時」輸出PDF文件,並處理不同的語言。與原始腳本如HTML2FPDF相比,它的速度較慢,並且在使用Unicode字體時產生的檔案較大,但支援CSS樣式等,並進行了大量增強。支援幾乎所有語言,包括RTL(阿拉伯語和希伯來語)和CJK(中日韓)。支援嵌套的區塊級元素(如P、DIV),

VSCode Windows 64位元 下載

VSCode Windows 64位元 下載

微軟推出的免費、功能強大的一款IDE編輯器