成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

如何手寫一個線程池?

開發 開發工具
按需創建線程,不要一開始就創建「corePoolSize」個線程,而是在調用者提交任務的過程中逐漸創建出來,最后創建了「corePoolSize」個就不再創建了。

手寫一個異步工具類

我是小識,新來了一個公司。這個公司呢,有個特點,就是很鼓勵大家封裝各種實用的工具類,提高開發效率。

于是我就到處看項目的源碼,看看有沒有什么能改進的?果然讓我發現了。項目中到處充斥著 new Thread 類來異步執行代碼的邏輯

new Thread(r).start();

我們可以封裝一個異步工具類啊!

第一版

說干就干,把上面的代碼簡單封裝一下,一個簡單的異步工具類就封裝好了

public interface Executor {

void execute(Runnable r);
}
public class AsyncExecutorV1 implements Executor {

@Override
public void execute(Runnable r) {
new Thread(r).start();
}
}

于是開開心心的提交了 merge request。

第二版

正當我滿懷期待工具類代碼能被合并的時候,沒想代碼被組長杰哥打回來了。

「杰哥」:有心封裝工具類值得鼓勵,不過還可以改進一下

「小識」:還能再改進?沒感覺我這個工具類還有改進的余地啊!

「杰哥」:假如說有10000個異步任務,你這創建10000個線程,資源耗費太嚴重了

「小識」:這樣啊,那我加個隊列,任務都放到隊列中,用一個線程從隊列中取任務執行

public class AsyncExecutorV2 implements Executor {

private BlockingQueue<Runnable> workQueue;

public AsyncExecutorV2(BlockingQueue<Runnable> workQueue) {
this.workQueue = workQueue;
WorkThread workThread = new WorkThread();
workThread.start();
}

@SneakyThrows
@Override
public void execute(Runnable r) {
workQueue.add(r);
}

class WorkThread extends Thread {

@Override
public void run() {
while (true) {
Runnable task = null;
try {
task = workQueue.take();
} catch (InterruptedException e) {
e.printStackTrace();
}
task.run();
}
}
}
}

第三版

「小識」:杰哥,快幫我看看,還有啥改進的沒?

「杰哥」:小伙子不錯啊,居然能想到用隊列來緩沖任務,不愧是我招進來的人!但是用一個異步線程執行任務,你確定這個工具類比同步執行的效率快?

「小識」:哈哈,又一個工具類翻車的案例,應該多開幾個異步線程來執行任務,但是應該開多少呢?

「杰哥」:誰最清楚異步工具類應該用多少個線程來執行呢?

「小識」:使用工具類的人。

「杰哥」:這不對了,你可以定義一個線程數量參數,讓用戶來決定開多少線程。「另外你這個工具類還個問題,隊列滿了會直接拋出異常!」

「小識」:那我增加一個拒絕策略類(RejectedExecutionHandler),當線程池滿了讓用戶決定執行策略,比如直接拋異常,用當前線程同步執行任務。

public class AsyncExecutorV3 implements Executor {

private BlockingQueue<Runnable> workQueue;

private List<WorkThread> workThreadList = new ArrayList<>();

private RejectedExecutionHandler handler;

public AsyncExecutorV3(int corePoolSize,
BlockingQueue<Runnable> workQueue,
RejectedExecutionHandler handler) {
this.workQueue = workQueue;
this.handler = handler;
for (int i = 0; i < corePoolSize; i++) {
WorkThread workThread = new WorkThread();
workThread.start();
workThreadList.add(workThread);
}
}

@SneakyThrows
@Override
public void execute(Runnable r) {
if (!workQueue.offer(r)) {
// 隊列滿了,執行拒絕策略
handler.rejectedExecution(r);
}
}

class WorkThread extends Thread {

@Override
public void run() {
while (true) {
Runnable task = null;
try {
task = workQueue.take();
} catch (InterruptedException e) {
e.printStackTrace();
}
task.run();
}
}
}
}
// 拒絕策略類
public interface RejectedExecutionHandler {

void rejectedExecution(Runnable r);
}
// 當線程池滿了之后直接拋出異常
public class AbortPolicy implements RejectedExecutionHandler {

@Override
public void rejectedExecution(Runnable r) {
throw new RuntimeException("queue is full");
}
}
// 當線程池滿了之后直接拋出異常
public class AbortPolicy implements RejectedExecutionHandler {

@Override
public void rejectedExecution(Runnable r) {
throw new RuntimeException("queue is full");
}
}
// 當線程池滿了之后,用提交任務的線程同步執行任務
public class CallerRunsPolicy implements RejectedExecutionHandler {

@Override
public void rejectedExecution(Runnable r) {
r.run();
}
}

再次提交 merge request,終于被合并了,別的團隊都開始使用我的工具類了!

過了幾天小亮急匆匆找到我。

「小亮」:小識,你的工具類挺好用的。但是我最近遇到了一個問題,我用了CountDownLatch批量執行任務,但是我這個任務好像卡住了,我用jstack想看看線程的執行情況,快告訴我你異步線程的名字設置的是啥?

「小識」:哎呀,我們沒設置線程的名字,應該用的是默認的線程名字 Thread-n。

「小亮」:你可得給工具類加個線程名字的參數啊,不然一個一個看線程的狀態太累了,而且效率也不高

「小識」:我這就加

第四版

趕緊加了一個線程名字的參數,然后再次提交代碼

「杰哥」:哎呀,沒想到我也疏忽了,沒發現這個問題,確實應該加個線程名字的參數,代碼的可擴展性太重要了,改來改去可不行

「小識」:是啊!

「杰哥」:你覺得你只加一個線程名字參數,可擴展性高嗎?如果有的團隊想修改異步線程的優先級,你再加個優先級參數?

「小識」:感覺不太行,那讓用戶把線程傳給我吧!

「杰哥」:哈哈,可以,你還可以用工廠模式優化一下,用戶傳入線程工廠類,工具類用工廠類創建線程

「小識」:不愧是杰哥,這樣一來代碼更清爽了!

public class AsyncExecutorV4 implements Executor {

private BlockingQueue<Runnable> workQueue;

private List<WorkThread> workThreadList = new ArrayList<>();

private RejectedExecutionHandler handler;

public AsyncExecutorV4(int corePoolSize,
BlockingQueue<Runnable> workQueue,
RejectedExecutionHandler handler,
ThreadFactory threadFactory) {
this.workQueue = workQueue;
this.handler = handler;
for (int i = 0; i < corePoolSize; i++) {
// 用工廠類創建線程
WorkThread workThread = threadFactory.newThread();
workThread.start();
workThreadList.add(workThread);
}
}

@SneakyThrows
@Override
public void execute(Runnable r) {
if (!workQueue.offer(r)) {
handler.rejectedExecution(r);
}
}

// 異步線程
public class WorkThread extends Thread {

@Override
public void run() {
while (true) {
Runnable task = null;
try {
task = workQueue.take();
} catch (InterruptedException e) {
e.printStackTrace();
}
task.run();
}
}
}

// 異步線程工廠類
public interface ThreadFactory {
WorkThread newThread();
}
}

代碼提交之后,小亮給線程起了一個名字,async-thread,現在他通過名字很快就能知道線程池中的線程在干嘛了!

大家不斷的進行改進

隨著這個異步工具類在公司內部使用的越來越多,大家也提交了很多改進的代碼。

  • 按需創建線程,不要一開始就創建「corePoolSize」個線程,而是在調用者提交任務的過程中逐漸創建出來,最后創建了「corePoolSize」個就不再創建了
  • 提高工具的彈性,當任務突增時,隊列會被放滿,然后多余的任務有可能會被直接扔掉。當然我們可以把「corePoolSize」設的很大,但是這樣并不優雅,因為大部分情況下是用不到這么多線程的。當任務突增時,我們可以適當增加線程,提高執行速度,當然創建的總線程數還是要限制一下的,我們把能創建的總數定為「maximumPoolSize」
  • 及時關閉不需要的線程,當任務突增時,線程數可能增加「maximumPoolSize」,但是大多數時間「corePoolSize」個線程就足夠用了,因此可以定義一個超時時間,當一個線程在「keepAliveTime」時間內沒有執行任務,就把它給關掉

異步工具類執行流程圖

經過大家的不斷改進之后,構造函數中的參數也越來越多了,杰哥讓我寫個文檔吧,把這個異步工具類的構造函數和執行流程總結一下,不然新來的小伙伴看到這個工具類一臉懵可不行!

這個工具類的構造函數目前有如下7個參數

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

參數

含義

corePoolSize

核心線程數

maximumPoolSize

最大線程數

keepAliveTime

非核心線程的空閑時間

TimeUnit

空閑時間的單位

BlockingQueue<Runnable>

任務隊列

ThreadFactory

線程工廠

RejectedExecutionHandler

拒絕策略

「執行流程圖如下」:

對了,最后大家給這個異步工具類起了一個牛的名字,「線程池」。

責任編輯:武曉燕 來源: Java識堂
相關推薦

2020-10-27 13:24:35

線程池系統模型

2020-11-02 08:19:18

RPC框架Java

2021-03-18 08:04:54

AQS工具CAS

2017-03-02 13:31:02

監控系統

2020-12-10 08:24:40

線程池線程方法

2025-06-26 03:12:00

2021-03-29 08:47:24

線程面試官線程池

2021-11-15 10:35:46

Python線程代碼

2021-12-28 15:10:01

線程池C語言編程語言

2023-10-16 22:03:36

日志包多線程日志包

2021-10-27 06:49:34

線程池Core函數

2024-08-02 09:49:35

Spring流程Tomcat

2021-05-27 09:50:03

連接池FTP服務器

2021-12-07 06:55:17

節流函數Throttle

2021-04-18 07:12:08

Dubbo線程池

2022-02-10 11:43:54

DUBBO線程池QPS

2023-03-08 07:43:07

DUCC配置平臺

2022-01-26 15:20:00

配置微服務架構

2021-02-22 17:17:38

Proxy緩存代碼

2022-12-30 08:29:07

Nacos動態化線程池
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 日韩av中文| 欧美一级片免费看 | 激情五月婷婷综合 | 国产精品观看 | 97超在线视频 | 中文字幕一区二区三区精彩视频 | 一区二区免费看 | 日韩久久精品 | 国产精品电影在线观看 | 久热伊人 | 亚洲欧美成人影院 | 麻豆av在线 | 精品国产亚洲一区二区三区大结局 | 久久精品电影 | 羞羞视频免费在线观看 | 国产a视频 | 91视频在线观看 | 国产精品明星裸体写真集 | av中文字幕在线播放 | 噜久寡妇噜噜久久寡妇 | 国产精品免费在线 | 日韩av福利在线观看 | 久久久一区二区三区 | 久久综合av | 久久精品日产第一区二区三区 | 天堂在线中文字幕 | 欧美精品一区在线发布 | 国产一级久久久久 | 中文字幕成人网 | 成人性视频在线 | 亚洲一区二区在线视频 | 中文字幕第二十页 | 日韩小视频 | 久久亚洲天堂 | 精精国产xxxx视频在线播放 | 成人黄色av网址 | 成人超碰 | 久久久久亚洲国产| 亚洲国产成人在线 | 人人爽人人草 | 青青久草 |