10分钟带你徒手做个Java线程池

摘要:花10分钟开发一个极简版的Java线程池,让小伙伴们更好的理解线程池的核心原理。

本文分享自华为云社区《放大招了,冰河带你10分钟手撸Java线程池,yyds,赶快收藏吧》,作者:冰 河。

Java线程池核心原理

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

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

各参数的含义如下所示。

  • corePoolSize:线程池中的常驻核心线程数。
  • maximumPoolSize:线程池能够容纳同时执行的最大线程数,此值大于等于1。
  • keepAliveTime:多余的空闲线程存活时间,当空间时间达到keepAliveTime值时,多余的线程会被销毁直到只剩下corePoolSize个线程为止。
  • unit:keepAliveTime的单位。
  • workQueue:任务队列,被提交但尚未被执行的任务。
  • threadFactory:表示生成线程池中工作线程的线程工厂,用户创建新线程,一般用默认即可。
  • handler:拒绝策略,表示当线程队列满了并且工作线程大于等于线程池的最大显示数(maxnumPoolSize)时,如何来拒绝请求执行的runnable的策略。

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

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

10分钟带你徒手做个Java线程池

手撸Java线程池

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

10分钟带你徒手做个Java线程池

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

10分钟带你徒手做个Java线程池

定义核心字段

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

  • DEFAULT_WORKQUEUE_SIZE:静态常量,表示默认的阻塞队列大小。
  • workQueue:模拟实际的线程池使用阻塞队列来实现生产者-消费者模式。
  • workThreads:模拟实际的线程池使用List集合保存线程池内部的工作线程。

核心代码如下所示。

//默认阻塞队列大小
private static final int DEFAULT_WORKQUEUE_SIZE = 5;
//模拟实际的线程池使用阻塞队列来实现生产者-消费者模式
private BlockingQueue<Runnable> workQueue;
//模拟实际的线程池使用List集合保存线程池内部的工作线程
private List<WorkThread> workThreads = new ArrayList<WorkThread>();

创建内部类WordThread

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

核心代码如下所示。

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

创建ThreadPool类的构造方法

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

核心代码如下所示。

//在ThreadPool的构造方法中传入线程池的大小和阻塞队列
public ThreadPool(int poolSize, BlockingQueue<Runnable> workQueue){
 this.workQueue = workQueue;
 //创建poolSize个工作线程并将其加入到workThreads集合中
 IntStream.range(0, poolSize).forEach((i) -> {
 WorkThread workThread = new WorkThread();
 workThread.start();
 workThreads.add(workThread);
 });
}
//在ThreadPool的构造方法中传入线程池的大小
public ThreadPool(int poolSize){
 this(poolSize, new LinkedBlockingQueue<>(DEFAULT_WORKQUEUE_SIZE));
}

创建执行任务的方法

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

核心代码如下所示。

//通过线程池执行任务
public void execute(Runnable task){
 try {
 workQueue.put(task);
 } catch (InterruptedException e) {
 e.printStackTrace();
 }
}

完整源码

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

package io.binghe.thread.pool;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.stream.IntStream;
/**
 * @author binghe
 * @version 1.0.0
 * @description 自定义线程池
 */
public class ThreadPool {
 //默认阻塞队列大小
 private static final int DEFAULT_WORKQUEUE_SIZE = 5;
 //模拟实际的线程池使用阻塞队列来实现生产者-消费者模式
 private BlockingQueue<Runnable> workQueue;
 //模拟实际的线程池使用List集合保存线程池内部的工作线程
 private List<WorkThread> workThreads = new ArrayList<WorkThread>();
 //在ThreadPool的构造方法中传入线程池的大小和阻塞队列
 public ThreadPool(int poolSize, BlockingQueue<Runnable> workQueue){
 this.workQueue = workQueue;
 //创建poolSize个工作线程并将其加入到workThreads集合中
 IntStream.range(0, poolSize).forEach((i) -> {
 WorkThread workThread = new WorkThread();
 workThread.start();
 workThreads.add(workThread);
 });
 }
 //在ThreadPool的构造方法中传入线程池的大小
 public ThreadPool(int poolSize){
 this(poolSize, new LinkedBlockingQueue<>(DEFAULT_WORKQUEUE_SIZE));
 }
 //通过线程池执行任务
 public void execute(Runnable task){
 try {
 workQueue.put(task);
 } catch (InterruptedException e) {
 e.printStackTrace();
 }
 }
 //内部类WorkThread,模拟线程池中的工作线程
 //主要的作用就是消费workQueue中的任务,并执行
 //由于工作线程需要不断从workQueue中获取任务,使用了while(true)循环不断尝试消费队列中的任务
 class WorkThread extends Thread{
 @Override
 public void run() {
 //不断循环获取队列中的任务
 while (true){
 //当没有任务时,会阻塞
 try {
 Runnable workTask = workQueue.take();
 workTask.run();
 } catch (InterruptedException e) {
 e.printStackTrace();
 }
 }
 }
 }
}

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

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

编写测试程序

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

整体测试代码如下所示。

package io.binghe.thread.pool.test;
import io.binghe.thread.pool.ThreadPool;
import java.util.stream.IntStream;
/**
 * @author binghe
 * @version 1.0.0
 * @description 测试自定义线程池
 */
public class ThreadPoolTest {
 public static void main(String[] args){
 ThreadPool threadPool = new ThreadPool(10);
 IntStream.range(0, 10).forEach((i) -> {
 threadPool.execute(() -> {
 System.out.println(Thread.currentThread().getName() + "--->> Hello ThreadPool");
 });
 });
 }
}

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

Thread-0--->> Hello ThreadPool
Thread-9--->> Hello ThreadPool
Thread-5--->> Hello ThreadPool
Thread-8--->> Hello ThreadPool
Thread-4--->> Hello ThreadPool
Thread-1--->> Hello ThreadPool
Thread-2--->> Hello ThreadPool
Thread-5--->> Hello ThreadPool
Thread-9--->> Hello ThreadPool
Thread-0--->> Hello ThreadPool

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

总结

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

 

点击关注,第一时间了解华为云新鲜技术~

原文链接:https://www.cnblogs.com/huaweiyun/p/17333748.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:10分钟带你徒手做个Java线程池 - Python技术站

(0)
上一篇 2023年4月19日
下一篇 2023年4月20日

相关文章

  • Java LinkedList的实现原理图文详解

    首先,我们来了解一下Java LinkedList的基本特性。LinkedList是Java中实现链表数据结构的一种方式,它实现了List、Deque、Queue接口。LinkedList内部以链表的形式存储元素,每个节点都包含上一个节点的引用和下一个节点的引用。因此可以方便的在链表的任意位置进行添加、删除操作,但是随机访问某个元素的效率会比较低。 Link…

    Java 2023年5月26日
    00
  • Spring Boot使用模板引擎JSP实例解析

    针对“Spring Boot使用模板引擎JSP实例解析”的完整攻略,我将按照以下步骤逐一解析: 1. 添加依赖 首先,我们需要在pom.xml中添加JSP依赖。在<dependencies>标签内添加以下代码: <dependencies> <!– 省略其他依赖 … –> <dependency> &l…

    Java 2023年5月19日
    00
  • MyBatis 详细讲解动态 SQL的使用

    MyBatis 详细讲解动态 SQL的使用 MyBatis是一个支持动态SQL的持久层框架,可以使用简单的XML或注解进行配置。动态SQL是指能够在运行时根据不同条件生成不同SQL语句的能力。这种能力使我们能够构建出非常灵活的SQL语句,从而更好地满足项目需求。在本文中,我们将学习如何使用MyBatis的动态SQL。 1. if 标签 if 标签用来在满足一…

    Java 2023年5月20日
    00
  • java基于数据库实现全局唯一ID的示例

    以下是“java基于数据库实现全局唯一ID的示例”的完整攻略及两条示例: 一、前置条件 在进行本教程之前,请确保以下条件已经满足: 你已熟悉Java编程语言,并且能够独立编写Java代码; 你已经安装了MySQL数据库,并掌握了基本操作; 你已经安装了Java开发环境和相关依赖库。 二、方案选择 目前常见的实现全局唯一ID的方案有雪花算法、UUID等。本教程…

    Java 2023年5月20日
    00
  • Spring MVC配置双数据源实现一个java项目同时连接两个数据库的方法

    要在Spring MVC中配置双数据源来连接两个数据库,需要以下步骤: 添加数据库连接的相关依赖 需要在pom.xml文件中添加数据库的相关依赖,例如: <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boo…

    Java 2023年5月20日
    00
  • Java检查日期字符串是否合法的方法总结

    下面是详细的讲解。 一、问题描述 在Java中,经常需要对日期字符串进行处理。然而,在处理日期字符串时,会遇到日期格式不正确的情况。因此,如何检查一个日期字符串是否符合某种固定格式是非常重要的。 二、实现思路 检查一个日期字符串是否合法的主要思路是对日期格式进行校验。Java中提供了许多日期格式的校验方式,比较常用的有以下几种: 使用SimpleDateFo…

    Java 2023年5月20日
    00
  • Spring Boot 整合 Apache Dubbo的示例代码

    这里给出一个完整的 Spring Boot 整合 Apache Dubbo 的示例代码攻略,包含以下内容: 环境准备 创建 Spring Boot 项目并添加依赖 配置 Dubbo 的注册中心和提供者 编写 Dubbo 的服务提供者 编写 Dubbo 的服务消费者 运行并测试示例代码 以下是具体的步骤: 1. 环境准备 首先,你需要安装并配置好以下环境: J…

    Java 2023年5月19日
    00
  • Hibernate 与 Mybatis 的共存问题,打破你的认知!(两个ORM框架)

    Hibernate 与 Mybatis 的共存问题,打破你的认知!(两个ORM框架) 背景介绍 Hibernate 和 Mybatis 都是 Java 中常用的 ORM 框架,可以用来操作数据库。相比较于传统的 JDBC 操作数据库,ORM 框架具备更高的抽象性和易用性。Hibernate 和 Mybatis 都有其自身的特点和优势,因此在一些情况下,我们需…

    Java 2023年5月20日
    00
合作推广
合作推广
分享本页
返回顶部