java 多线程之Worker Thread模式(Thread Pool模式)
Worker Thread模式
Worker的意思是工作的人,在Worker Thread模式中,工人线程Worker thread会逐个取回工作并进行处理,当所有工作全部完成后,工人线程会等待新的工作到来。
Worker Thread模式也被成为Background Thread(背景线程)模式,另外,如果从保存多个工人线程的场所这一点看,我们也可以称这种模式为Thread Pool模式。
Worker Thread模式中的角色
TransPortThread [将部件塞到传送带上](委托者)
创建表示工作请求的Cargo并将其传递给SimpleChannel 。在示例程序中,TransPortThread 相当于该角色。
2.SimpleChannel [传送带的角色](通信线路)
SimpleChannel 角色接受来自于TransPortThread 的Cargo,并将其传递给WorkerThread。在示例程序中,SimpleChannel 相当于该角色。
3.WorkerThread(工人)
WorkerThread角色从Channel中获取Cargo,并进行工作,当一项工作完成后,它会继续去获取另外的Cargo,在示例程序中,WorkerThread相当于该角色。
4.Cargo(部件)
Cargo角色是表示工作中的部件,Cargo角色中保存了进行工作所必须的信息,在示例程序中,Cargo相当于该部件角色。
Worker Thread使用场景
想象这样一个场景,一个工厂在生产货物,在一个车间里,有几个工人,每次生产部件准备后,车间外的人就将部件放到车间的传送带上,工人每次做完一个货品就从桌子上取部件。在这里,注意到,部件并不是直接交给工人的,另外一点,工人并不是做完一个部件就回家换个新人,后者在现实有点滑稽,但是在程序中却对应一个典型的线程使用方法:线程池。
所谓线程池,就是对线程的复用,当线程执行完任务之后就继续取其他任务执行,而不是销毁启动新线程执行其他任务。因为线程的启动对于系统性能开销比较大,所以这样对于系统性能的提高很有好处。
Main.java
public class WorkClient {public static void main(String[] args) {// 创建容量为5的SimpleChannel实例(数据缓冲通道)SimpleChannel simpleChannel = new SimpleChannel(5);// 启动工作通道开始处理数据(消费者启动)simpleChannel.startWork();// 启动三个数据传输线程(生产者角色)new TransPortThread("june", simpleChannel).start();new TransPortThread("july", simpleChannel).start();new TransPortThread("alen", simpleChannel).start();}}
Cargo.java
角色:部件或货物
import java.util.Optional;/*** 货物类,包含货物编号和名称,并提供执行操作时打印线程执行信息的功能** @author [请填写作者名称]* @version 1.0*/
public class Cargo {/*** 货物名称*/private final String name;/*** 货物编号*/private final int num;/*** 构造方法,初始化货物信息** @param name 货物名称* @param num 货物编号*/public Cargo(String name, int num) {this.name = name;this.num = num;}/*** 执行货物处理操作,打印当前线程名称和货物信息*/public void execute(){Optional.of(Thread.currentThread().getName()+" executed "+this.toString()).ifPresent(System.out::println);}/*** 获取货物信息的字符串表示** @return 格式化后的货物信息(例如:Cargo ==> NO.123 Name.ItemA)*/@Overridepublic String toString() {return "Cargo ==> NO."+num+" Name."+name;}
}
SimpleChannel.java
职责:将部件或货物从源头传给处理端
import java.util.Arrays;/*** 简单线程通信通道,管理货物队列和工作线程池* 使用环形缓冲区实现生产者-消费者模式,通过synchronized和wait/notify机制保证线程安全*/
public class SimpleChannel {/** 缓冲区最大容量 */private static final int BUFFER_SIZE = 100;/** 环形缓冲区数组,存储货物对象 */private final Cargo[] cargoQueue;/** 队列头指针,指向下一个要取出的元素位置 */private int headIndex;/** 当前队列中的元素数量 */private int count;/** 队列尾指针,指向下一个要插入的元素位置 */private int tailIndex;/** 工作线程池 */private final WorkerThread[] workerPool;/*** 构造方法初始化通道* @param workerCount 工作线程数量*/public SimpleChannel(int workerCount) {this.cargoQueue = new Cargo[BUFFER_SIZE];this.headIndex = 0;this.count = 0;this.tailIndex = 0;this.workerPool = new WorkerThread[workerCount];this.init();}/*** 初始化工作线程池* 为每个线程设置名称并绑定当前通道实例*/private void init() {for (int i = 0; i < workerPool.length; i++) {workerPool[i] = new WorkerThread("worker-"+i, this);}}/*** 启动所有工作线程开始工作* 通过遍历线程池调用start()方法启动每个线程*/public void startWork() {Arrays.asList(workerPool).forEach(WorkerThread::start);}/*** 生产者添加货物到缓冲区* @param cargo 要添加的货物对象* @throws InterruptedException 线程中断异常*/public synchronized void pushCargo(Cargo cargo) {while (count >= BUFFER_SIZE) {try {this.wait(); // 缓冲区满时等待} catch (InterruptedException e) {// 异常处理(建议补充日志)}}// 添加货物到尾部并更新指针this.cargoQueue[tailIndex] = cargo;this.tailIndex = (this.tailIndex + 1) % this.cargoQueue.length;this.count++;this.notifyAll(); // 通知等待线程}/*** 消费者从缓冲区取出货物* @return 取出的货物对象* @throws InterruptedException 线程中断异常*/public synchronized Cargo popCargo() {while (count <= 0) {try {this.wait(); // 缓冲区空时等待} catch (InterruptedException e) {// 异常处理(建议补充日志)}}// 取出头部元素并更新指针Cargo cargo = this.cargoQueue[headIndex];this.headIndex = (this.headIndex + 1) % this.cargoQueue.length;this.count--;this.notifyAll(); // 通知等待线程return cargo;}
}
TransPortThread .java
职责:将请求的信息塞到传输通道上去
/*** 运输线程,负责持续生成货物并推送到传输通道 {@link SimpleChannel}*/
public class TransPortThread extends Thread {private SimpleChannel simpleChannel;/*** 静态随机数生成器,用于生成随机休眠时间*/private static final Random RANDOM = new Random(System.currentTimeMillis());/*** 构造方法** @param name 线程名称* @param simpleChannel 货物传输通道*/public TransPortThread(String name, SimpleChannel simpleChannel) {super(name);this.simpleChannel = simpleChannel;}@Overridepublic void run() {try {int i = 0;// 无限循环持续生成货物while (true){Cargo cargo = new Cargo(getName(), i++); // 创建货物对象(包含线程名和序号)this.simpleChannel.pushCargo(cargo); // 将货物推送到通道Thread.sleep(RANDOM.nextInt(1000)); // 随机休眠 0-1000ms}} catch (InterruptedException e) {// 线程中断时的异常处理(当前空实现)}}
}
WorkerThread.java
职责:从传输带上获取部件或货物
import java.util.Random;
/*** 工作线程类,负责从共享通道中持续获取任务并执行*/
public class WorkerThread extends Thread {/*** 存储任务通道实例,用于获取待处理的任务对象*/private final SimpleChannel simpleChannel;/*** 静态随机数生成器,用于生成随机休眠时间(模拟任务执行间隔)*/private static final Random random = new Random(System.currentTimeMillis());/*** 构造方法初始化线程名称和任务通道* @param name 线程名称标识* @param simpleChannel 任务通道对象,用于获取待处理任务*/public WorkerThread(String name, SimpleChannel simpleChannel) {super(name);this.simpleChannel = simpleChannel;}/*** 线程执行主体方法,持续循环执行以下操作:* 1. 从通道中获取任务对象* 2. 执行任务* 3. 随机休眠(模拟任务处理间隔)*/@Overridepublic void run() {while (true) {try {// 从通道中获取任务对象Cargo cargo = simpleChannel.popCargo();// 执行任务逻辑cargo.execute();// 随机休眠0-1000毫秒(模拟任务间隔)Thread.sleep(random.nextInt(1000));} catch (Exception e) {// 异常处理:打印堆栈信息(实际应用中建议添加更完善的异常处理)e.printStackTrace();}}}
}
执行的结果
worker-1 executed Cargo ==> NO.0 Name.alen
worker-0 executed Cargo ==> NO.0 Name.june
worker-4 executed Cargo ==> NO.0 Name.july
worker-3 executed Cargo ==> NO.1 Name.june
worker-0 executed Cargo ==> NO.1 Name.alen
worker-2 executed Cargo ==> NO.1 Name.july
worker-1 executed Cargo ==> NO.2 Name.june
worker-0 executed Cargo ==> NO.2 Name.july
worker-2 executed Cargo ==> NO.2 Name.alen
worker-3 executed Cargo ==> NO.3 Name.june
worker-4 executed Cargo ==> NO.4 Name.june
worker-2 executed Cargo ==> NO.3 Name.alen
worker-3 executed Cargo ==> NO.3 Name.july
worker-4 executed Cargo ==> NO.5 Name.june
worker-1 executed Cargo ==> NO.6 Name.june
worker-0 executed Cargo ==> NO.4 Name.alen
worker-2 executed Cargo ==> NO.7 Name.june
worker-3 executed Cargo ==> NO.5 Name.alen
worker-2 executed Cargo ==> NO.4 Name.july
worker-4 executed Cargo ==> NO.6 Name.alen
worker-0 executed Cargo ==> NO.5 Name.july
worker-2 executed Cargo ==> NO.8 Name.june
worker-3 executed Cargo ==> NO.7 Name.alen
worker-1 executed Cargo ==> NO.6 Name.july
worker-4 executed Cargo ==> NO.8 Name.alen
worker-4 executed Cargo ==> NO.9 Name.june
worker-2 executed Cargo ==> NO.10 Name.june
worker-3 executed Cargo ==> NO.9 Name.alen
worker-1 executed Cargo ==> NO.7 Name.july
worker-2 executed Cargo ==> NO.10 Name.alen
worker-4 executed Cargo ==> NO.11 Name.alen
worker-3 executed Cargo ==> NO.11 Name.june
worker-0 executed Cargo ==> NO.8 Name.july
worker-4 executed Cargo ==> NO.12 Name.alen
worker-3 executed Cargo ==> NO.12 Name.june
worker-4 executed Cargo ==> NO.13 Name.alen
worker-1 executed Cargo ==> NO.9 Name.july
worker-2 executed Cargo ==> NO.14 Name.alen
worker-0 executed Cargo ==> NO.13 Name.june
worker-4 executed Cargo ==> NO.15 Name.alen
worker-2 executed Cargo ==> NO.10 Name.july
worker-1 executed Cargo ==> NO.14 Name.june
worker-3 executed Cargo ==> NO.15 Name.june
worker-4 executed Cargo ==> NO.16 Name.alen
worker-3 executed Cargo ==> NO.11 Name.july
worker-2 executed Cargo ==> NO.12 Name.july
worker-0 executed Cargo ==> NO.16 Name.june
worker-1 executed Cargo ==> NO.17 Name.june
worker-2 executed Cargo ==> NO.18 Name.june
worker-2 executed Cargo ==> NO.17 Name.alen
worker-4 executed Cargo ==> NO.13 Name.july
worker-0 executed Cargo ==> NO.18 Name.alen
worker-0 executed Cargo ==> NO.19 Name.june
worker-3 executed Cargo ==> NO.14 Name.july
worker-1 executed Cargo ==> NO.19 Name.alen
worker-2 executed Cargo ==> NO.20 Name.june
worker-3 executed Cargo ==> NO.21 Name.june
worker-0 executed Cargo ==> NO.15 Name.july
worker-3 executed Cargo ==> NO.20 Name.alen
worker-4 executed Cargo ==> NO.21 Name.alen
worker-3 executed Cargo ==> NO.22 Name.june
worker-1 executed Cargo ==> NO.16 Name.july
worker-2 executed Cargo ==> NO.17 Name.july
worker-0 executed Cargo ==> NO.23 Name.june
相关文章:
java 多线程之Worker Thread模式(Thread Pool模式)
Worker Thread模式 Worker的意思是工作的人,在Worker Thread模式中,工人线程Worker thread会逐个取回工作并进行处理,当所有工作全部完成后,工人线程会等待新的工作到来。 Worker Thread模式也被成为Background Threadÿ…...
在 MoonBit 中引入 Elm 架构:用简单原则打造健壮的 Web 应用
Elm 是一种纯函数式编程语言,专为构建前端 Web 应用程序而设计。它编译为 JavaScript,强调简洁性、性能和健壮性。 纯函数式的含义是函数没有副作用,这使得代码更易于理解和调试。通过强大的静态类型检查,Elm 确保应用程序不会抛…...
CMD命令行笔记
CMD命令行笔记,涵盖常用命令及实用技巧,适合快速查阅: 一、基础操作 打开CMD Win R → 输入 cmd → 回车管理员模式:右键开始菜单 → 选择“命令提示符(管理员)” 常用命令 help:查看所有命令…...
Python自动化办公
第五篇:Python自动化办公:10行代码搞定重复性工作 适合读者:职场人士、数据分析师 | 阅读时长:12分钟 引言 每天重复处理Excel、PDF或邮件?Python可以帮你自动化这些枯燥任务,节省90%的时间。本文通过实际…...
PDF 转换为 Word、HTML、LaTeX 和 Markdown 格式
PDF 转换为 Word、HTML、LaTeX 和 Markdown 格式 1. Doc2XReferences https://doc2x.com/ 1. Doc2X References [1] Yongqiang Cheng, https://yongqiang.blog.csdn.net/ [2] GPT 学术优化 (GPT Academic), https://github.com/binary-husky/gpt_academic [3] 学术版 GPT 网页…...
C#中async await异步关键字用法和异步的底层原理
目录 C#异步编程一、异步编程基础二、异步方法的工作原理三、代码示例四、编译后的底层实现五、总结 C#异步编程 一、异步编程基础 异步编程是啥玩意儿 就是让程序在干等着某些耗时操作(比如等网络响应、读写文件啥的)的时候,能把线程腾出来…...
shardingsphere-jdbc集成Seata分布式事务
1、导入相关依赖 <!-- shardingsphere-jdbc --><dependency><groupId>org.apache.shardingsphere</groupId><artifactId>shardingsphere-jdbc</artifactId><version>5.5.1</version></dependency><!-- shardingspher…...
华为OD机试真题——统计匹配的二元组个数(2025A卷:100分)Java/python/JavaScript/C++/C语言/GO六种最佳实现
2025 A卷 100分 题型 本文涵盖详细的问题分析、解题思路、代码实现、代码详解、测试用例以及综合分析; 并提供Java、python、JavaScript、C、C语言、GO六种语言的最佳实现方式! 2025华为OD真题目录全流程解析/备考攻略/经验分享 华为OD机试真题《统计匹配…...
CSS 表格样式学习笔记
CSS 提供了强大的工具来美化和定制 HTML 表格的外观。通过合理使用 CSS 属性,可以使表格更加美观、易读且功能强大。以下是对 CSS 表格样式的详细学习笔记。 一、表格边框 1. 单独边框 默认情况下,表格的 <table>、<th> 和 <td> 元…...
MySQL表的增删改查进阶版
Mysql 1、数据库的约束1.1约束类型1.2 NULL约束1.3 UNIQUE:唯一约束1.4 DEFAULT:默认值约束1.5 PRIMARY KEY:主键约束(重点)1.6 FOREIGN KEY:外键约束(重点) 2.表的设计2.1一对一2.2…...
记录 | Pycharm中如何调用Anaconda的虚拟环境
目录 前言一、步骤Step1 查看anaconda 环境名Step2 Python项目编译器更改 更新时间 前言 参考文章: 参考视频:如何在pycharm中使用Anaconda创建的python环境 自己的感想 这里使用的Pycharm 2024专业版的。我所使用的Pycharm专业版位置:【仅用…...
2025年K8s最新高频面试题
目录 Kubernetes的核心组件有哪些,各自作用是什么? Pod和Deployment的区别? Service有哪些类型,分别适用于什么场景? ConfigMap和Secret有什么区别? StatefulSet 和 Deployment 的主要区别是什么? 什么是 Ingress,有哪些常用实现方式? 如何限制 Kubernetes 中 Pod …...
【Android】LiveData深度解析
一,概述 1,LiveData是状态订阅组件,是粘性的,而非事件订阅组件(可以没有事件,但不能没有状态)。所谓的状态,即UI状态,同一时刻只存在一种,且是最新状态,过期的状态应该被遗弃。事件,则是生产者创建的事件,需一一消费,不能被遗弃。 2,Android页面承载组件Activ…...
数据结构专题 - 线性表
线性表是数据结构中最基础、最常用的数据结构之一,它在实际应用中非常广泛。无论是操作系统中的内存管理,还是数据库中的索引结构,线性表都扮演着重要角色。 一、线性表的概念与抽象数据类型 1.1 线性表的逻辑结构 线性表是由n(…...
上门送水小程序区域代理模块框架设计
一、逻辑分析 代理申请流程: 潜在代理商通过小程序提交代理申请,需要填写个人或企业基本信息、联系方式、期望代理区域等。系统收到申请后,进行初步审核,检查信息的完整性和合规性。运营人员进行人工审核,根据公司政策…...
asp-for等常用的HTML辅助标记?
在ASP.NET Core Razor Pages 和 MVC 中,除了asp-for之外,还有许多常用的 HTML 辅助标记,下面为你详细介绍: 表单与路由相关 asp-action 和 asp-controller 用途:这两个标记用于生成表单或链接的 URL,指定…...
qt pyqt5的开发, 修改psd图像
这是引子, 需要将这个 photoshop-python-api 进行使用 https://juejin.cn/post/7445112318693621797#heading-4 这个是ps-python-api的官网, 在里面找api文档 https://pypi.org/project/photoshop-python-api/ 源码.gitee.url https://gitee.com/lbnb/psd_work.git 一. 安装必要…...
Spring 中的循环依赖问题:解决方案与三级缓存机制
目录 Spring 中的循环依赖问题:解决方案与三级缓存机制什么是循环依赖?循环依赖的定义循环依赖的举例 Spring 中的循环依赖类型1. 构造器注入引发的循环依赖2. Setter 注入引发的循环依赖3. 字段注入(Autowired)引发的循环依赖 Sp…...
ios接入穿山甲【Swift】
1.可接入的广告,点击右下角查看接入文档 https://www.csjplatform.com/union/media/union/download/groMore 2.进入接入文档,选择最新版本进行接入 pod Ads-CN-Beta,6.8.0.2pod GMGdtAdapter-Beta, 4.15.22.0pod GDTMobSDK,4.15.30pod KSAdSDK,3.3.74.0p…...
蓝桥杯大模板
init.c void System_Init() {P0 0x00; //关闭蜂鸣器和继电器P2 P2 & 0x1f | 0xa0;P2 & 0x1f;P0 0x00; //关闭LEDP2 P2 & 0x1f | 0x80;P2 & 0x1f; } led.c #include <LED.H>idata unsigned char temp_1 0x00; idata unsigned char temp_old…...
电脑一直不关机会怎么样?电脑长时间不关机的影响
现代生活中,许多人会让自己的电脑24小时不间断运行,无论是为了持续的工作、娱乐,还是出于忘记关机的习惯。然而,电脑长时间不关机,除了提供便利之外,也可能对设备的健康产生一系列影响。本文将为大家介绍电…...
vue3 当页面显示了 p/span/div 标签 想要转换成正常文字
返回值有标签出现时,使用v-html 解决 <p>{{ item.content }}</p> //页面直接显示接口返回的带标签的数据 <p v-html"item.content "></p> //转换成html文件 显示正常文字各种样式 问题: 解决:v-html 显…...
Elasticsearch 8.18 中提供了原生连接 (Native Joins)
作者:来自 Elastic Costin Leau 探索 LOOKUP JOIN,这是一条在 Elasticsearch 8.18 的技术预览中提供的新 ES|QL 命令。 很高兴宣布 LOOKUP JOIN —— 这是一条在 Elasticsearch 8.18 的技术预览中提供的新 ES|QL 命令,旨在执行左 joins 以进行…...
java CountDownLatch用法简介
CountDownLatch倒计数锁存器 CountDownLatch:用于协同控制一个或多个线程等待在其他线程中执行的一组操作完成,然后再继续执行 CountDownLatch用法 构造方法:CountDownLatch(int count),count指定等待的条件数(任务…...
k8s蓝绿发布
k8s蓝绿发布 什么是蓝绿部署K8S中如何实现蓝绿部署k8s蓝绿部署流程图 什么是蓝绿部署 参考: https://youtu.be/CLq_hA0lAd0 https://help.coding.net/docs/cd/best-practice/blue-green.html 蓝绿部署最早是由马丁福勒 2010年在他的博客中提出. 蓝绿部署是一种软件部署策略,用…...
链接世界:计算机网络的核心与前沿
计算机网络引言 在数字化时代,计算机网络已经成为我们日常生活和工作中不可或缺的基础设施。从简单的局域网(LAN)到全球互联网,计算机网络将数以亿计的设备连接在一起,推动了信息交换、资源共享以及全球化的进程。 什…...
记录Docker部署CosyVoice V2.0声音克隆
#记录工作 CosyVoice 是由 FunAudioLLM 团队开发的一个开源多语言大规模语音生成模型,提供了从推理、训练到部署的全栈解决方案。 项目地址: https://github.com/FunAudioLLM/CosyVoice.git 该项目目前从v1.0版本迭代到v2.0版本,但是在Wind…...
MCU刷写——HEX与S19文件互转详解及Python实现
工作之余来写写关于MCU的Bootloader刷写的相关知识,以免忘记。今天就来聊聊Hex与S19这这两种文件互相转化,我是分享人M哥,目前从事车载控制器的软件开发及测试工作。 学习过程中如有任何疑问,可底下评论! 如果觉得文章内容在工作学习中有帮助到你,麻烦点赞收藏评论+关注走…...
全链路开源数据平台技术选型指南:六大实战工具链解析
在数字化转型加速的背景下,开源技术正重塑数据平台的技术格局。本文深度解析数据平台的全链路架构,精选六款兼具创新性与实用性的开源工具,涵盖数据编排、治理、实时计算、联邦查询等核心场景,为企业构建云原生数据架构提供可落地…...
C++学习:六个月从基础到就业——面向对象编程:封装、继承与多态
C学习:六个月从基础到就业——面向对象编程:封装、继承与多态 本文是我C学习之旅系列的第九篇技术文章,主要讨论C中面向对象编程的三大核心特性:封装、继承与多态。这些概念是理解和应用面向对象设计的关键。查看完整系列目录了解…...
