0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

Java线程池核心原理

OSC开源社区 来源:冰河技术 2023-04-21 10:24 次阅读
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

本文的整体结构如下所示。

f7ed04c8-df9c-11ed-bfe3-dac502259ad0.png

Java线程池核心原理

看过Java线程池源码的小伙伴都知道,在Java线程池中最核心的类就是ThreadPoolExecutor,而在ThreadPoolExecutor类中最核心的构造方法就是带有7个参数的构造方法,如下所示。

publicThreadPoolExecutor(intcorePoolSize,
intmaximumPoolSize,
longkeepAliveTime,
TimeUnitunit,
BlockingQueueworkQueue,
ThreadFactorythreadFactory,
RejectedExecutionHandlerhandler)

各参数的含义如下所示。

corePoolSize:线程池中的常驻核心线程数。

maximumPoolSize:线程池能够容纳同时执行的最大线程数,此值大于等于1。

keepAliveTime:多余的空闲线程存活时间,当空间时间达到keepAliveTime值时,多余的线程会被销毁直到只剩下corePoolSize个线程为止。

unit:keepAliveTime的单位。

workQueue:任务队列,被提交但尚未被执行的任务。

threadFactory:表示生成线程池中工作线程的线程工厂,用户创建新线程,一般用默认即可。

handler:拒绝策略,表示当线程队列满了并且工作线程大于等于线程池的最大显示数(maxnumPoolSize)时,如何来拒绝请求执行的runnable的策略。

并且Java的线程池是通过 生产者-消费者模式 实现的,线程池的使用方是生产者,而线程池本身就是消费者。

Java线程池的核心工作流程如下图所示。

f8050b90-df9c-11ed-bfe3-dac502259ad0.png

手撸Java线程池

我们自己手动实现的线程池要比Java自身的线程池简单的多,我们去掉了各种复杂的处理方式,只保留了最核心的原理:线程池的使用者向任务队列中添加任务,而线程池本身从任务队列中消费任务并执行任务。

f812d59a-df9c-11ed-bfe3-dac502259ad0.png

只要理解了这个核心原理,接下来的代码就简单多了。在实现这个简单的线程池时,我们可以将整个实现过程进行拆解。拆解后的实现流程为:定义核心字段、创建内部类WorkThread、创建ThreadPool类的构造方法和创建执行任务的方法。

f81fd010-df9c-11ed-bfe3-dac502259ad0.png

定义核心字段

首先,我们创建一个名称为ThreadPool的Java类,并在这个类中定义如下核心字段。

DEFAULT_WORKQUEUE_SIZE:静态常量,表示默认的阻塞队列大小。

workQueue:模拟实际的线程池使用阻塞队列来实现生产者-消费者模式。

workThreads:模拟实际的线程池使用List集合保存线程池内部的工作线程。

核心代码如下所示。

//默认阻塞队列大小
privatestaticfinalintDEFAULT_WORKQUEUE_SIZE=5;

//模拟实际的线程池使用阻塞队列来实现生产者-消费者模式
privateBlockingQueueworkQueue;

//模拟实际的线程池使用List集合保存线程池内部的工作线程
privateListworkThreads=newArrayList();

创建内部类WordThread

在ThreadPool类中创建一个内部类WorkThread,模拟线程池中的工作线程。主要的作用就是消费workQueue中的任务,并执行任务。由于工作线程需要不断从workQueue中获取任务,所以,这里使用了while(true)循环不断尝试消费队列中的任务。

核心代码如下所示。

//内部类WorkThread,模拟线程池中的工作线程
//主要的作用就是消费workQueue中的任务,并执行
//由于工作线程需要不断从workQueue中获取任务,使用了while(true)循环不断尝试消费队列中的任务
classWorkThreadextendsThread{
@Override
publicvoidrun(){
//不断循环获取队列中的任务
while(true){
//当没有任务时,会阻塞
try{
RunnableworkTask=workQueue.take();
workTask.run();
}catch(InterruptedExceptione){
e.printStackTrace();
}
}
}
}

创建ThreadPool类的构造方法

这里,我们为ThreadPool类创建两个构造方法,一个构造方法中传入线程池的容量大小和阻塞队列,另一个构造方法中只传入线程池的容量大小。

核心代码如下所示。

//在ThreadPool的构造方法中传入线程池的大小和阻塞队列
publicThreadPool(intpoolSize,BlockingQueueworkQueue){
this.workQueue=workQueue;
//创建poolSize个工作线程并将其加入到workThreads集合中
IntStream.range(0,poolSize).forEach((i)->{
WorkThreadworkThread=newWorkThread();
workThread.start();
workThreads.add(workThread);
});
}

//在ThreadPool的构造方法中传入线程池的大小
publicThreadPool(intpoolSize){
this(poolSize,newLinkedBlockingQueue<>(DEFAULT_WORKQUEUE_SIZE));
}

创建执行任务的方法

在ThreadPool类中创建执行任务的方法execute(),execute()方法的实现比较简单,就是将方法接收到的Runnable任务加入到workQueue队列中。

核心代码如下所示。

//通过线程池执行任务
publicvoidexecute(Runnabletask){
try{
workQueue.put(task);
}catch(InterruptedExceptione){
e.printStackTrace();
}
}

完整源码

这里,我们给出手动实现的ThreadPool线程池的完整源代码,如下所示。

packageio.binghe.thread.pool;

importjava.util.ArrayList;
importjava.util.List;
importjava.util.concurrent.BlockingQueue;
importjava.util.concurrent.LinkedBlockingQueue;
importjava.util.stream.IntStream;

/**
*@authorbinghe
*@version1.0.0
*@description自定义线程池
*/
publicclassThreadPool{

//默认阻塞队列大小
privatestaticfinalintDEFAULT_WORKQUEUE_SIZE=5;

//模拟实际的线程池使用阻塞队列来实现生产者-消费者模式
privateBlockingQueueworkQueue;

//模拟实际的线程池使用List集合保存线程池内部的工作线程
privateListworkThreads=newArrayList();

//在ThreadPool的构造方法中传入线程池的大小和阻塞队列
publicThreadPool(intpoolSize,BlockingQueueworkQueue){
this.workQueue=workQueue;
//创建poolSize个工作线程并将其加入到workThreads集合中
IntStream.range(0,poolSize).forEach((i)->{
WorkThreadworkThread=newWorkThread();
workThread.start();
workThreads.add(workThread);
});
}

//在ThreadPool的构造方法中传入线程池的大小
publicThreadPool(intpoolSize){
this(poolSize,newLinkedBlockingQueue<>(DEFAULT_WORKQUEUE_SIZE));
}

//通过线程池执行任务
publicvoidexecute(Runnabletask){
try{
workQueue.put(task);
}catch(InterruptedExceptione){
e.printStackTrace();
}
}

//内部类WorkThread,模拟线程池中的工作线程
//主要的作用就是消费workQueue中的任务,并执行
//由于工作线程需要不断从workQueue中获取任务,使用了while(true)循环不断尝试消费队列中的任务
classWorkThreadextendsThread{
@Override
publicvoidrun(){
//不断循环获取队列中的任务
while(true){
//当没有任务时,会阻塞
try{
RunnableworkTask=workQueue.take();
workTask.run();
}catch(InterruptedExceptione){
e.printStackTrace();
}
}
}
}
}

没错,我们仅仅用了几十行Java代码就实现了一个极简版的Java线程池,没错,这个极简版的Java线程池的代码却体现了Java线程池的核心原理。

接下来,我们测试下这个极简版的Java线程池。

编写测试程序

测试程序也比较简单,就是通过在main()方法中调用ThreadPool类的构造方法,传入线程池的大小,创建一个ThreadPool类的实例,然后循环10次调用ThreadPool类的execute()方法,向线程池中提交的任务为:打印当前线程的名称--->> Hello ThreadPool。

整体测试代码如下所示。

packageio.binghe.thread.pool.test;

importio.binghe.thread.pool.ThreadPool;

importjava.util.stream.IntStream;

/**
*@authorbinghe
*@version1.0.0
*@description测试自定义线程池
*/
publicclassThreadPoolTest{

publicstaticvoidmain(String[]args){
ThreadPoolthreadPool=newThreadPool(10);
IntStream.range(0,10).forEach((i)->{
threadPool.execute(()->{
System.out.println(Thread.currentThread().getName()+"--->>HelloThreadPool");
});
});
}
}

接下来,运行ThreadPoolTest类的main()方法,会输出如下信息。

Thread-0--->>HelloThreadPool
Thread-9--->>HelloThreadPool
Thread-5--->>HelloThreadPool
Thread-8--->>HelloThreadPool
Thread-4--->>HelloThreadPool
Thread-1--->>HelloThreadPool
Thread-2--->>HelloThreadPool
Thread-5--->>HelloThreadPool
Thread-9--->>HelloThreadPool
Thread-0--->>HelloThreadPool

至此,我们自定义的Java线程池就开发完成了。

总结

线程池的核心原理其实并不复杂,只要我们耐心的分析,深入其源码理解线程池的核心本质,你就会发现线程池的设计原来是如此的优雅。希望通过这个手写线程池的小例子,能够让你更好的理解线程池的核心原理。





审核编辑:刘清

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • JAVA
    +关注

    关注

    20

    文章

    3008

    浏览量

    116842
  • 线程池
    +关注

    关注

    0

    文章

    58

    浏览量

    7426

原文标题:10分钟带你徒手做个Java线程池

文章出处:【微信号:OSC开源社区,微信公众号:OSC开源社区】欢迎添加关注!文章转载请注明出处。

收藏 人收藏
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

    评论

    相关推荐
    热点推荐

    Java并发编程的“基石”——多线程概念初识

    Java 得以切入的核心契机。 二、 隐形基石:Java 并发体系在调度中的降维打击 很多人对 Java 的印象停留在 Web 开发,却忽略了 J
    发表于 04-16 18:50

    动态流量数据资源交付技术:破解网络流量调度的核心难题

    前言在数字经济全面渗透的当下,数据传输与网络流量已成为支撑数字业务运行的核心基础。从日常的互联网访问、企业跨地域办公,到工业物联网终端互联、智算中心的算力调度,都离不开稳定、高效的流量资源交付体系
    的头像 发表于 03-30 13:50 1880次阅读
    动态流量<b class='flag-5'>池</b>数据资源交付技术:破解网络流量调度的<b class='flag-5'>核心</b>难题

    生产环境数据库连接耗尽的全流程排查与性能优化实战

    数据库连接是应用程序与数据库之间的缓存连接组件。连接在应用程序启动时创建一组数据库连接,应用程序从连接获取连接,使用完毕后归还连接而非关闭连接,避免反复建立和关闭连接的性能开销
    的头像 发表于 03-27 15:58 328次阅读

    全栈国产AI Coding上线:摩尔线程+硅基流动+智谱,强强联合!

    摩尔线程今日正式推出AI Coding Plan 智能编程服务。作为首个基于国产全功能 GPU 算力底座构建的智能开发解决方案,该服务以 MTT S5000 强劲的全精度计算能力为核心驱动,融合硅
    的头像 发表于 02-03 17:07 1976次阅读
    全栈国产AI Coding上线:摩尔<b class='flag-5'>线程</b>+硅基流动+智谱,强强联合!

    解析Linux的进程、线程和协程

    和系统资源。线程的引入使得多核处理器得以充分利用,因为多线程程序可以更有效地分配和管理多核心的计算资源。 线程的特点包括: (1)共享性:线程
    发表于 12-22 11:00

    开放协作,共筑生态——思尔芯参与上海开放处理器产业创新中心开业仪式暨RISC-V专利联盟专利仪式

    在推动芯片产业自主创新、构建开放协作生态的背景下,上海开放处理器产业创新中心(SOPIC)于2025年12月12日在上海张江正式举办开业仪式。同期举行的RISC-V专利仪式上,思尔芯作为首批入
    的头像 发表于 12-15 17:38 1303次阅读
    开放协作,共筑生态——思尔芯参与上海开放处理器产业创新中心开业仪式暨RISC-V专利联盟专利<b class='flag-5'>池</b>入<b class='flag-5'>池</b>仪式

    C语言内存使用

    ,整个堆有可能被弄得支离破碎,最终导致大量内存浪费。 那么这种情况下,我们解决这类问题的思路,就是创建一个内存。 内存,实际上就是我们让程序创建出来的一块额外的缓存区域,如果有需要释放内存,先
    发表于 12-11 07:57

    线程的系统

    线程系统的事件响应也是在中断中完成的,但事件的处理是在线程中完成的。在多线程系统中,线程跟中断一样,也具有优先级,优先级高的线程会被优先执
    发表于 12-08 07:55

    Linux多线程对比单线程的优势

    在Linux系统中,线程是操作系统能够进行运算调度的最小单位。线程被包含在进程之中,是进程中的实际运行单位。一个进程可以拥有多个线程,这些线程共享相同的内存空间和系统资源。
    发表于 12-01 06:11

    Arm Neoverse CPU上大代码量Java应用的性能测试

    Java 是互联网领域广泛使用的编程语言。Java 应用的一些特性使其性能表现与提前编译的原生应用(例如 C 程序)大相径庭。由于 Java 字节码无法直接在 CPU 上执行,因此通常运行时在
    的头像 发表于 11-05 11:25 946次阅读
    Arm Neoverse CPU上大代码量<b class='flag-5'>Java</b>应用的性能测试

    数据全复用高性能化层设计思路分享

    大家好,本团队此次分享的内容为可实现数据全复用高性能化层设计思路,核心部分主要由以下3个部分组成; 1.SRAM读取模块;——化使用的存储为SRAM 基于SRAM读与写时序,约束
    发表于 10-29 07:10

    沉砂物联网监控管理系统方案

    沉砂作为污水处理工艺中的核心预处理单元,承担着去除污水中砂粒、砾石等重型颗粒物的关键任务。其运行效率直接影响后续处理设备的寿命与整体处理效果。随着环保政策趋严与污水处理规模扩大,传统人工巡检模式已
    的头像 发表于 10-14 14:08 572次阅读
    沉砂<b class='flag-5'>池</b>物联网监控管理系统方案

    Java效率提升指南:5个Java工具选型建议及Perforce JRebel和XRebel介绍

    企业级Java环境越来越复杂,真正的破局点,可能不在“人”,而在于“工具”。5个实用建议,帮你理清Java工具的选型思路。
    的头像 发表于 09-11 13:59 1923次阅读
    <b class='flag-5'>Java</b>效率提升指南:5个<b class='flag-5'>Java</b>工具选型建议及Perforce JRebel和XRebel介绍

    摩尔线程“AI工厂”:五大核心技术支撑,打造大模型训练超级工厂

    2025年7月25日,上海——在世界人工智能大会(WAIC 2025)开幕前夕,摩尔线程以“算力进化,精度革命”为主题举办技术分享会,并创新性提出“AI工厂”理念。摩尔线程创始人兼CEO张建中在主题
    的头像 发表于 07-28 11:28 4849次阅读
    摩尔<b class='flag-5'>线程</b>“AI工厂”:五大<b class='flag-5'>核心</b>技术支撑,打造大模型训练超级工厂

    鸿蒙5开发宝藏案例分享---应用并发设计

    线程 --> |系统托管| FFRT_I/O[FFRT I/O线程] 三大核心理念 : 内存隔离 :
    发表于 06-12 16:19