文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

10分钟手撸Java线程池,yyds!!

2024-12-02 19:32

关注

最近有不少小伙伴私信我说:看了我在【精通高并发系列】文章中写的深度解析线程池源码部分的文章,但是还是有些不明白线程池的实现原理。问我能不能手写一个简单的线程池,帮助读者深刻理解线程池的原理。

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

在【精通高并发系列】的文章中,我们曾经深度解析过线程池的源码,从源码层面深度解析了线程池的实现原理。

其实,源码是原理落地的最直接体现,看懂源码对于深刻理解原理有着很大的帮助。但是不少小伙伴看源码时,总觉得源码太枯燥了,看不懂。

那今天,我们就一起花10分钟手撸一个极简版的Java线程池,让小伙伴们更好的理解线程池的核心原理。

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

Java线程池核心原理

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

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

各参数的含义如下所示。

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

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

手撸Java线程池

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

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

定义核心字段

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

核心代码如下所示。

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

创建内部类WordThread

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

核心代码如下所示。

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

创建ThreadPool类的构造方法

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

核心代码如下所示。

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

创建执行任务的方法

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

核心代码如下所示。

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

完整源码

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

  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. public class ThreadPool { 
  11.  
  12.     //默认阻塞队列大小 
  13.     private static final int DEFAULT_WORKQUEUE_SIZE = 5; 
  14.  
  15.     //模拟实际的线程池使用阻塞队列来实现生产者-消费者模式 
  16.     private BlockingQueue workQueue; 
  17.  
  18.     //模拟实际的线程池使用List集合保存线程池内部的工作线程 
  19.     private List workThreads = new ArrayList(); 
  20.  
  21.     //在ThreadPool的构造方法中传入线程池的大小和阻塞队列 
  22.     public ThreadPool(int poolSize, BlockingQueue workQueue){ 
  23.         this.workQueue = workQueue; 
  24.         //创建poolSize个工作线程并将其加入到workThreads集合中 
  25.         IntStream.range(0, poolSize).forEach((i) -> { 
  26.             WorkThread workThread = new WorkThread(); 
  27.             workThread.start(); 
  28.             workThreads.add(workThread); 
  29.         }); 
  30.     } 
  31.  
  32.     //在ThreadPool的构造方法中传入线程池的大小 
  33.     public ThreadPool(int poolSize){ 
  34.         this(poolSize, new LinkedBlockingQueue<>(DEFAULT_WORKQUEUE_SIZE)); 
  35.     } 
  36.  
  37.  //通过线程池执行任务 
  38.     public void execute(Runnable task){ 
  39.         try { 
  40.             workQueue.put(task); 
  41.         } catch (InterruptedException e) { 
  42.             e.printStackTrace(); 
  43.         } 
  44.     } 
  45.  
  46.     //内部类WorkThread,模拟线程池中的工作线程 
  47.     //主要的作用就是消费workQueue中的任务,并执行 
  48.     //由于工作线程需要不断从workQueue中获取任务,使用了while(true)循环不断尝试消费队列中的任务 
  49.     class WorkThread extends Thread{ 
  50.         @Override 
  51.         public void run() { 
  52.             //不断循环获取队列中的任务 
  53.             while (true){ 
  54.                 //当没有任务时,会阻塞 
  55.                 try { 
  56.                     Runnable workTask = workQueue.take(); 
  57.                     workTask.run(); 
  58.                 } catch (InterruptedException e) { 
  59.                     e.printStackTrace(); 
  60.                 } 
  61.             } 
  62.         } 
  63.     } 

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

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

编写测试程序

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

整体测试代码如下所示。

  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. public class ThreadPoolTest { 
  9.  
  10.     public static void main(String[] args){ 
  11.         ThreadPool threadPool = new ThreadPool(10); 
  12.         IntStream.range(0, 10).forEach((i) -> { 
  13.             threadPool.execute(() -> { 
  14.                 System.out.println(Thread.currentThread().getName() + "--->> Hello ThreadPool"); 
  15.             }); 
  16.         }); 
  17.     } 

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

  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线程池就开发完成了。

总结

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

本文转载自微信公众号「冰河技术」,可以通过以下二维码关注。转载本文请联系冰河技术公众号。

 

来源:冰河技术内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯