自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

10分鐘手?jǐn)]Java線(xiàn)程池,yyds??!

開(kāi)發(fā) 后端
那今天,我們就一起花10分鐘手?jǐn)]一個(gè)極簡(jiǎn)版的Java線(xiàn)程池,讓小伙伴們更好的理解線(xiàn)程池的核心原理。

[[428225]]

最近有不少小伙伴私信我說(shuō):看了我在【精通高并發(fā)系列】文章中寫(xiě)的深度解析線(xiàn)程池源碼部分的文章,但是還是有些不明白線(xiàn)程池的實(shí)現(xiàn)原理。問(wèn)我能不能手寫(xiě)一個(gè)簡(jiǎn)單的線(xiàn)程池,幫助讀者深刻理解線(xiàn)程池的原理。

這不,我熬夜肝了這篇文章。

在【精通高并發(fā)系列】的文章中,我們?cè)?jīng)深度解析過(guò)線(xiàn)程池的源碼,從源碼層面深度解析了線(xiàn)程池的實(shí)現(xiàn)原理。

其實(shí),源碼是原理落地的最直接體現(xiàn),看懂源碼對(duì)于深刻理解原理有著很大的幫助。但是不少小伙伴看源碼時(shí),總覺(jué)得源碼太枯燥了,看不懂。

那今天,我們就一起花10分鐘手?jǐn)]一個(gè)極簡(jiǎn)版的Java線(xiàn)程池,讓小伙伴們更好的理解線(xiàn)程池的核心原理。

本文的整體結(jié)構(gòu)如下所示。

Java線(xiàn)程池核心原理

看過(guò)Java線(xiàn)程池源碼的小伙伴都知道,在Java線(xiàn)程池中最核心的類(lèi)就是ThreadPoolExecutor,而在ThreadPoolExecutor類(lèi)中最核心的構(gòu)造方法就是帶有7個(gè)參數(shù)的構(gòu)造方法,如下所示。

  1. public ThreadPoolExecutor(int corePoolSize, 
  2.                              int maximumPoolSize, 
  3.                              long keepAliveTime, 
  4.                              TimeUnit unit, 
  5.                              BlockingQueue<Runnable> workQueue, 
  6.                              ThreadFactory threadFactory, 
  7.                              RejectedExecutionHandler handler) 

各參數(shù)的含義如下所示。

  • corePoolSize:線(xiàn)程池中的常駐核心線(xiàn)程數(shù)。
  • maximumPoolSize:線(xiàn)程池能夠容納同時(shí)執(zhí)行的最大線(xiàn)程數(shù),此值大于等于1。
  • keepAliveTime:多余的空閑線(xiàn)程存活時(shí)間,當(dāng)空間時(shí)間達(dá)到keepAliveTime值時(shí),多余的線(xiàn)程會(huì)被銷(xiāo)毀直到只剩下corePoolSize個(gè)線(xiàn)程為止。
  • unit:keepAliveTime的單位。
  • workQueue:任務(wù)隊(duì)列,被提交但尚未被執(zhí)行的任務(wù)。
  • threadFactory:表示生成線(xiàn)程池中工作線(xiàn)程的線(xiàn)程工廠(chǎng),用戶(hù)創(chuàng)建新線(xiàn)程,一般用默認(rèn)即可。
  • handler:拒絕策略,表示當(dāng)線(xiàn)程隊(duì)列滿(mǎn)了并且工作線(xiàn)程大于等于線(xiàn)程池的最大顯示數(shù)(maxnumPoolSize)時(shí),如何來(lái)拒絕請(qǐng)求執(zhí)行的runnable的策略。

并且Java的線(xiàn)程池是通過(guò) 生產(chǎn)者-消費(fèi)者模式 實(shí)現(xiàn)的,線(xiàn)程池的使用方是生產(chǎn)者,而線(xiàn)程池本身就是消費(fèi)者。

Java線(xiàn)程池的核心工作流程如下圖所示。

手?jǐn)]Java線(xiàn)程池

我們自己手動(dòng)實(shí)現(xiàn)的線(xiàn)程池要比Java自身的線(xiàn)程池簡(jiǎn)單的多,我們?nèi)サ袅烁鞣N復(fù)雜的處理方式,只保留了最核心的原理:線(xiàn)程池的使用者向任務(wù)隊(duì)列中添加任務(wù),而線(xiàn)程池本身從任務(wù)隊(duì)列中消費(fèi)任務(wù)并執(zhí)行任務(wù)。

只要理解了這個(gè)核心原理,接下來(lái)的代碼就簡(jiǎn)單多了。在實(shí)現(xiàn)這個(gè)簡(jiǎn)單的線(xiàn)程池時(shí),我們可以將整個(gè)實(shí)現(xiàn)過(guò)程進(jìn)行拆解。拆解后的實(shí)現(xiàn)流程為:定義核心字段、創(chuàng)建內(nèi)部類(lèi)WorkThread、創(chuàng)建ThreadPool類(lèi)的構(gòu)造方法和創(chuàng)建執(zhí)行任務(wù)的方法。

定義核心字段

首先,我們創(chuàng)建一個(gè)名稱(chēng)為T(mén)hreadPool的Java類(lèi),并在這個(gè)類(lèi)中定義如下核心字段。

  • DEFAULT_WORKQUEUE_SIZE:靜態(tài)常量,表示默認(rèn)的阻塞隊(duì)列大小。
  • workQueue:模擬實(shí)際的線(xiàn)程池使用阻塞隊(duì)列來(lái)實(shí)現(xiàn)生產(chǎn)者-消費(fèi)者模式。
  • workThreads:模擬實(shí)際的線(xiàn)程池使用List集合保存線(xiàn)程池內(nèi)部的工作線(xiàn)程。

核心代碼如下所示。

  1. //默認(rèn)阻塞隊(duì)列大小 
  2. private static final int DEFAULT_WORKQUEUE_SIZE = 5; 
  3.  
  4. //模擬實(shí)際的線(xiàn)程池使用阻塞隊(duì)列來(lái)實(shí)現(xiàn)生產(chǎn)者-消費(fèi)者模式 
  5. private BlockingQueue<Runnable> workQueue; 
  6.  
  7. //模擬實(shí)際的線(xiàn)程池使用List集合保存線(xiàn)程池內(nèi)部的工作線(xiàn)程 
  8. private List<WorkThread> workThreads = new ArrayList<WorkThread>(); 

創(chuàng)建內(nèi)部類(lèi)WordThread

在ThreadPool類(lèi)中創(chuàng)建一個(gè)內(nèi)部類(lèi)WorkThread,模擬線(xiàn)程池中的工作線(xiàn)程。主要的作用就是消費(fèi)workQueue中的任務(wù),并執(zhí)行任務(wù)。由于工作線(xiàn)程需要不斷從workQueue中獲取任務(wù),所以,這里使用了while(true)循環(huán)不斷嘗試消費(fèi)隊(duì)列中的任務(wù)。

核心代碼如下所示。

  1. //內(nèi)部類(lèi)WorkThread,模擬線(xiàn)程池中的工作線(xiàn)程 
  2. //主要的作用就是消費(fèi)workQueue中的任務(wù),并執(zhí)行 
  3. //由于工作線(xiàn)程需要不斷從workQueue中獲取任務(wù),使用了while(true)循環(huán)不斷嘗試消費(fèi)隊(duì)列中的任務(wù) 
  4. class WorkThread extends Thread{ 
  5.     @Override 
  6.     public void run() { 
  7.         //不斷循環(huán)獲取隊(duì)列中的任務(wù) 
  8.         while (true){ 
  9.             //當(dāng)沒(méi)有任務(wù)時(shí),會(huì)阻塞 
  10.             try { 
  11.                 Runnable workTask = workQueue.take(); 
  12.                 workTask.run(); 
  13.             } catch (InterruptedException e) { 
  14.                 e.printStackTrace(); 
  15.             } 
  16.         } 
  17.     } 

創(chuàng)建ThreadPool類(lèi)的構(gòu)造方法

這里,我們?yōu)門(mén)hreadPool類(lèi)創(chuàng)建兩個(gè)構(gòu)造方法,一個(gè)構(gòu)造方法中傳入線(xiàn)程池的容量大小和阻塞隊(duì)列,另一個(gè)構(gòu)造方法中只傳入線(xiàn)程池的容量大小。

核心代碼如下所示。

  1. //在ThreadPool的構(gòu)造方法中傳入線(xiàn)程池的大小和阻塞隊(duì)列 
  2. public ThreadPool(int poolSize, BlockingQueue<Runnable> workQueue){ 
  3.     this.workQueue = workQueue; 
  4.     //創(chuàng)建poolSize個(gè)工作線(xiàn)程并將其加入到workThreads集合中 
  5.     IntStream.range(0, poolSize).forEach((i) -> { 
  6.         WorkThread workThread = new WorkThread(); 
  7.         workThread.start(); 
  8.         workThreads.add(workThread); 
  9.     }); 
  10.  
  11. //在ThreadPool的構(gòu)造方法中傳入線(xiàn)程池的大小 
  12. public ThreadPool(int poolSize){ 
  13.     this(poolSize, new LinkedBlockingQueue<>(DEFAULT_WORKQUEUE_SIZE)); 

創(chuàng)建執(zhí)行任務(wù)的方法

在ThreadPool類(lèi)中創(chuàng)建執(zhí)行任務(wù)的方法execute(),execute()方法的實(shí)現(xiàn)比較簡(jiǎn)單,就是將方法接收到的Runnable任務(wù)加入到workQueue隊(duì)列中。

核心代碼如下所示。

  1. //通過(guò)線(xiàn)程池執(zhí)行任務(wù) 
  2. public void execute(Runnable task){ 
  3.     try { 
  4.         workQueue.put(task); 
  5.     } catch (InterruptedException e) { 
  6.         e.printStackTrace(); 
  7.     } 

完整源碼

這里,我們給出手動(dòng)實(shí)現(xiàn)的ThreadPool線(xiàn)程池的完整源代碼,如下所示。

  1. package io.binghe.thread.pool; 
  2.  
  3. import java.util.ArrayList; 
  4. import java.util.List; 
  5. import java.util.concurrent.BlockingQueue; 
  6. import java.util.concurrent.LinkedBlockingQueue; 
  7. import java.util.stream.IntStream; 
  8.  
  9. /** 
  10.  * @author binghe 
  11.  * @version 1.0.0 
  12.  * @description 自定義線(xiàn)程池 
  13.  */ 
  14. public class ThreadPool { 
  15.  
  16.     //默認(rèn)阻塞隊(duì)列大小 
  17.     private static final int DEFAULT_WORKQUEUE_SIZE = 5; 
  18.  
  19.     //模擬實(shí)際的線(xiàn)程池使用阻塞隊(duì)列來(lái)實(shí)現(xiàn)生產(chǎn)者-消費(fèi)者模式 
  20.     private BlockingQueue<Runnable> workQueue; 
  21.  
  22.     //模擬實(shí)際的線(xiàn)程池使用List集合保存線(xiàn)程池內(nèi)部的工作線(xiàn)程 
  23.     private List<WorkThread> workThreads = new ArrayList<WorkThread>(); 
  24.  
  25.     //在ThreadPool的構(gòu)造方法中傳入線(xiàn)程池的大小和阻塞隊(duì)列 
  26.     public ThreadPool(int poolSize, BlockingQueue<Runnable> workQueue){ 
  27.         this.workQueue = workQueue; 
  28.         //創(chuàng)建poolSize個(gè)工作線(xiàn)程并將其加入到workThreads集合中 
  29.         IntStream.range(0, poolSize).forEach((i) -> { 
  30.             WorkThread workThread = new WorkThread(); 
  31.             workThread.start(); 
  32.             workThreads.add(workThread); 
  33.         }); 
  34.     } 
  35.  
  36.     //在ThreadPool的構(gòu)造方法中傳入線(xiàn)程池的大小 
  37.     public ThreadPool(int poolSize){ 
  38.         this(poolSize, new LinkedBlockingQueue<>(DEFAULT_WORKQUEUE_SIZE)); 
  39.     } 
  40.  
  41.  //通過(guò)線(xiàn)程池執(zhí)行任務(wù) 
  42.     public void execute(Runnable task){ 
  43.         try { 
  44.             workQueue.put(task); 
  45.         } catch (InterruptedException e) { 
  46.             e.printStackTrace(); 
  47.         } 
  48.     } 
  49.  
  50.     //內(nèi)部類(lèi)WorkThread,模擬線(xiàn)程池中的工作線(xiàn)程 
  51.     //主要的作用就是消費(fèi)workQueue中的任務(wù),并執(zhí)行 
  52.     //由于工作線(xiàn)程需要不斷從workQueue中獲取任務(wù),使用了while(true)循環(huán)不斷嘗試消費(fèi)隊(duì)列中的任務(wù) 
  53.     class WorkThread extends Thread{ 
  54.         @Override 
  55.         public void run() { 
  56.             //不斷循環(huán)獲取隊(duì)列中的任務(wù) 
  57.             while (true){ 
  58.                 //當(dāng)沒(méi)有任務(wù)時(shí),會(huì)阻塞 
  59.                 try { 
  60.                     Runnable workTask = workQueue.take(); 
  61.                     workTask.run(); 
  62.                 } catch (InterruptedException e) { 
  63.                     e.printStackTrace(); 
  64.                 } 
  65.             } 
  66.         } 
  67.     } 

沒(méi)錯(cuò),我們僅僅用了幾十行Java代碼就實(shí)現(xiàn)了一個(gè)極簡(jiǎn)版的Java線(xiàn)程池,沒(méi)錯(cuò),這個(gè)極簡(jiǎn)版的Java線(xiàn)程池的代碼卻體現(xiàn)了Java線(xiàn)程池的核心原理。

接下來(lái),我們測(cè)試下這個(gè)極簡(jiǎn)版的Java線(xiàn)程池。

編寫(xiě)測(cè)試程序

測(cè)試程序也比較簡(jiǎn)單,就是通過(guò)在main()方法中調(diào)用ThreadPool類(lèi)的構(gòu)造方法,傳入線(xiàn)程池的大小,創(chuàng)建一個(gè)ThreadPool類(lèi)的實(shí)例,然后循環(huán)10次調(diào)用ThreadPool類(lèi)的execute()方法,向線(xiàn)程池中提交的任務(wù)為:打印當(dāng)前線(xiàn)程的名稱(chēng)--->> Hello ThreadPool。

整體測(cè)試代碼如下所示。

  1. package io.binghe.thread.pool.test; 
  2.  
  3. import io.binghe.thread.pool.ThreadPool; 
  4.  
  5. import java.util.stream.IntStream; 
  6.  
  7. /** 
  8.  * @author binghe 
  9.  * @version 1.0.0 
  10.  * @description 測(cè)試自定義線(xiàn)程池 
  11.  */ 
  12. public class ThreadPoolTest { 
  13.  
  14.     public static void main(String[] args){ 
  15.         ThreadPool threadPool = new ThreadPool(10); 
  16.         IntStream.range(0, 10).forEach((i) -> { 
  17.             threadPool.execute(() -> { 
  18.                 System.out.println(Thread.currentThread().getName() + "--->> Hello ThreadPool"); 
  19.             }); 
  20.         }); 
  21.     } 

接下來(lái),運(yùn)行ThreadPoolTest類(lèi)的main()方法,會(huì)輸出如下信息。

  1. Thread-0--->> Hello ThreadPool 
  2. Thread-9--->> Hello ThreadPool 
  3. Thread-5--->> Hello ThreadPool 
  4. Thread-8--->> Hello ThreadPool 
  5. Thread-4--->> Hello ThreadPool 
  6. Thread-1--->> Hello ThreadPool 
  7. Thread-2--->> Hello ThreadPool 
  8. Thread-5--->> Hello ThreadPool 
  9. Thread-9--->> Hello ThreadPool 
  10. Thread-0--->> Hello ThreadPool 

至此,我們自定義的Java線(xiàn)程池就開(kāi)發(fā)完成了。

總結(jié)

線(xiàn)程池的核心原理其實(shí)并不復(fù)雜,只要我們耐心的分析,深入其源碼理解線(xiàn)程池的核心本質(zhì),你就會(huì)發(fā)現(xiàn)線(xiàn)程池的設(shè)計(jì)原來(lái)是如此的優(yōu)雅。希望通過(guò)這個(gè)手寫(xiě)線(xiàn)程池的小例子,能夠讓你更好的理解線(xiàn)程池的核心原理。

本文轉(zhuǎn)載自微信公眾號(hào)「冰河技術(shù)」,可以通過(guò)以下二維碼關(guān)注。轉(zhuǎn)載本文請(qǐng)聯(lián)系冰河技術(shù)公眾號(hào)。

 

責(zé)任編輯:武曉燕 來(lái)源: 冰河技術(shù)
相關(guān)推薦

2021-05-31 07:22:46

ORM框架程序

2020-02-17 13:45:27

抓取代碼工具

2024-06-11 00:00:00

Java線(xiàn)程安全緩存組件

2020-10-29 08:28:42

Java NIO異步非阻塞

2021-07-15 06:43:11

Bash調(diào)試腳本

2013-09-13 14:08:01

2021-10-27 06:49:34

線(xiàn)程池Core函數(shù)

2023-12-27 08:15:47

Java虛擬線(xiàn)程

2021-04-23 09:50:41

topLinux命令

2014-08-08 09:30:04

android scrollview

2020-10-13 18:22:58

DevOps工具開(kāi)發(fā)

2018-06-26 05:23:19

線(xiàn)程安全函數(shù)代碼

2020-08-25 07:47:03

Java并發(fā)隊(duì)列

2022-08-29 15:02:53

遠(yuǎn)程桌面

2021-08-01 21:38:07

網(wǎng)頁(yè)點(diǎn)燈網(wǎng)關(guān)

2018-11-28 11:20:53

Python函數(shù)式編程編程語(yǔ)言

2021-05-28 07:38:20

內(nèi)存溢出場(chǎng)景

2021-05-17 20:13:50

數(shù)倉(cāng)操作型數(shù)據(jù)庫(kù)

2017-02-22 14:46:04

Leader-Foll線(xiàn)程

2019-07-18 16:32:06

Python函數(shù)數(shù)據(jù)
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)