Java多线程详解①①(全程干货!!!) 实现简单的线程池 || 定时器 || 简单实现定时器 || 时间轮实现定时器

这里是Themberfue
· 上一节讲了 线程池 线程池中的拒绝策略 等相关内容
实现简单的线程池
· 一个线程池最核心的方法就是 submit,通过 submit 提交 Runnable 任务来通知线程池来执行 Runnable 任务
· 我们简单实现一个特定线程数量的线程池,这些线程应该在线程池创建之初就被创建好,并不断尝试从任务队列中取出任务从而执行
· 所以还需一个阻塞队列用于存储 submit 提交过来的任务
· 代码实现:
import java.util.concurrent.ArrayBlockingQueue; import java.util.concurrent.BlockingQueue;/*** @author: Themberfue* @date: 2024/10/21 21:20* @description:*/ // 实现一个固定线程数的线程池 class MyThreadPool {private BlockingQueue<Runnable> queue = null;public MyThreadPool(int n) {// 初始化线程池,创建固定个数的线程// 这里使用ArrayBlockingQueue作为任务队列, 容量为1000this.queue = new ArrayBlockingQueue<>(1000);// 创建 n 个线程for (int i = 0; i < n; i++) {Thread t = new Thread(() -> {try {while (true) {Runnable take = queue.take();take.run();}} catch (InterruptedException e) {throw new RuntimeException(e);}});// 默认为前台线程,不会正常退出程序// t.setDaemon(true);t.start();}}public void submit(Runnable task) throws InterruptedException {// 将任务放入队列中queue.put(task);} }public class Demo34 {public static void main(String[] args) throws InterruptedException {MyThreadPool threadPool = new MyThreadPool(10);// 向线程池提交任务for (int i = 0; i < 100; i++) {int id = i;threadPool.submit(() -> {System.out.println("执行任务:" + id + " " + Thread.currentThread().getName());});}} }· 在执行完 100 个任务后,会发现程序并未结束,那是因为程序阻塞在了 queue.take() 中;况且线程池中的线程默认是前台线程,故不会随着主线程的结束而结束
· 在上一节的代码,看到了 shutdown 操作,也就是关闭线程池操作,该操作可以让线程池里的全部线程全部关闭,但不能保证线程池里的任务一定全部执行完毕;
· 除此之外,还提供了 awaitTermination 操作,该操作在线程池中的线程里的任务全部执行完毕后,再关闭线程池~
· 选择使用 shutdown 还是 awaitTermination 取决于任务的重要程度~
定时器
· 定时器类似于闹钟,时间到了就执行相关的逻辑~
· Java 标准库中提供了 Timer 类中的 schedule 方法来延迟执行某些逻辑~:
import java.util.Timer; import java.util.TimerTask;/*** @author: Themberfue* @date: 2024/10/22 19:24* @description:*/ public class Demo35 {public static void main(String[] args) {Timer timer = new Timer();// TimerTask是抽象类,但这里是通过创建 "匿名内部类" 的方式创建类timer.schedule(new TimerTask() {@Overridepublic void run() {System.out.println("3秒过后执行");}}, 3000);timer.schedule(new TimerTask() {@Overridepublic void run() {System.out.println("2秒过后执行");}}, 2000);timer.schedule(new TimerTask() {@Overridepublic void run() {System.out.println("1秒过后执行");}}, 1000);System.out.println("立即执行");} }· 与 Runnable 类似,这里描述任务的是 TimerTask 抽象类,通过创建一个匿名类继承了 TimerTask 类并且重写了 run 方法~
简单实现定时器
· 首先得有一个任务类TimerTask描述该任务逻辑
· 此外还需一个集合类来管理不同时间段要执行的任务;使用哪个集合类呢?必须得按照时间顺序执行任务,且遍历时间复杂度不能太高~ 没错,使用 优先级队列,可以快速找到哪个任务先执行
· 还需一个 schedule 方法来将任务添加到队列中
· 最后还需要额外创建一个线程来执行任务里的逻辑;此线程还需额外添加判断逻辑,与线程池的线程不同,该线程需要时间到了之后才可以执行相应的任务逻辑,时间没到就不执行,而不是直接执行
· 如何判断某任务到了时间需要执行?我们借助 时间戳 来完成这一功能,在创建 TimerTask 类时将当前 时间戳 与需要延迟执行的时间加上,循环判断当前时间戳是否达到加上后的结果;如果时间戳来到了这一结果,就说明可以执行该任务了
· 切记:既然要创建小根堆,我们需要规定该堆是按什么数值排序;在该案例中,应该是以执行该任务的延迟时间为基准,所以我们需要额外自定义一个比较器,用于小根堆的排序规则
import java.util.PriorityQueue;/*** @author: Themberfue* @date: 2024/10/22 19:32* @description:*/ class MyTimeTask implements Comparable<MyTimeTask>{// 需要执行的任务private final Runnable task;// 记录任务要执行的 "时刻"private final long time;public MyTimeTask(Runnable task, long time) {this.task = task;this.time = time;}public long getTime() {return time;}public void run() {task.run();}@Overridepublic int compareTo(MyTimeTask o) {return (int)(this.time - o.time);} }class MyTimer {PriorityQueue<MyTimeTask> queue = null;final Object locker = new Object();public MyTimer() {// 使用优先级队列(小根堆)来管理任务执行的顺序this.queue = new PriorityQueue<>();// 创建一个线程,负责执行队列中的任务Thread t = new Thread(() -> {try {while (true) {// 涉及多个线程,一定涉及线程安全问题synchronized (locker) {// 如果队列为空,那么让该线程等待,直到offer任务到队列中// 不推荐使用continue,避免反复快速执行while循环消耗不必要的资源while (queue.isEmpty()) {// continuelocker.wait();}MyTimeTask task = queue.peek();// 当前任务时间, 如果比系统时间大, 说明任务执行的时机未到if (System.currentTimeMillis() < task.getTime()) {// 避免使用 sleep 导致程序造成问题locker.wait( task.getTime() - System.currentTimeMillis());} else {// 时间到了,执行任务task.run();queue.poll();}}}} catch (InterruptedException e) {throw new RuntimeException();}});t.start();}public void schedule(Runnable task, long delay) {synchronized (locker) {MyTimeTask timeTask = new MyTimeTask(task, System.currentTimeMillis() + delay);queue.offer(timeTask);locker.notify();}} }public class Demo36 {public static void main(String[] args) {MyTimer myTimer = new MyTimer();myTimer.schedule(() -> {System.out.println("3秒后执行");}, 3000);myTimer.schedule(() -> {System.out.println("2秒后执行");}, 2000);myTimer.schedule(() -> {System.out.println("1秒后执行");}, 1000);System.out.println("立即执行");} }
· 上述代码中还有一些需要注意的点:
1. 调用 schedule 的是一个线程,定时器内部又有一个线程执行任务逻辑,那么就涉及到多线程操作,所以就可能存在线程安全问题,所以需要对程序进行加锁
2.
,这里的最初版本其实为:if (queue.isEmpty()) { continue; } ,那为什么改了呢?如果单纯为 if 判断是否空,且 continue 到循环首行,这里确实需要等,但这里却依旧在让 CPU 消耗资源,但并没有起到实质性的作用,说白了就是空等,CPU空转;为了优化代码,将其改为上述最终版本,使用 wait 操作,便可让 CPU 暂时对这里释放资源~~~
3.
这里可以换成 sleep 吗?当然不可以~;如果有一个程序允许时间为 11:00,有一个任务的运行时间为 11:30,此时 sleep 就会睡眠 30 分钟,若又有一个任务进来了,且它在 11:20 就要执行,但是 sleep 一旦触发,除非线程 Interrupt ,否则无法唤醒 sleep,那么就错过了后进来的任务的执行了,使用 wait,尽管是有参数版本的,依然可以被 notify 唤醒~~~
4. 在任务量少的场景下,一个线程执行这些操作绰绰有余,但如果碰上了非常大量的任务场景,此时一个线程就有点吃不消了,所以还是建议结合线程池来使用定时器,以防不必要的事情发生~~~
时间轮实现定时器
· 这里就是简单科普一下吧:交给 GPT 了~~~
设计思想
时间轮是一种环形数组结构,类似钟表的秒针。每个槽代表一个固定的时间单位(如 1 毫秒、1 秒等),每个槽可以存储多个定时任务。当时间推进时,指针在环上移动,指向当前时间对应的槽,并触发其中的任务。
执行流程
- 定义一个固定大小的环形数组(如 360 个槽,表示 360 秒)。
- 将定时任务根据触发时间计算分配到对应槽中:
- 任务剩余时间 < 一圈:分配到当前圈;
- 任务剩余时间 > 一圈:计圈数延迟,直到圈数耗尽。
- 定时器线程按固定频率移动指针,每次移动一格:
- 执行指针所在槽中的任务;
- 清空已完成任务。
优点
- 高效性:插入和触发任务的时间复杂度为 O(1)。
- 节省内存:通过时间轮的槽结构减少了对定时任务的单独存储。
- 适合大规模任务:当任务数量极多,触发时间分布较广时,时间轮比优先级队列效率高。
缺点
- 时间精度受限:时间粒度由槽的间隔决定,不能无限精确。
- 复杂度较高:需要处理多圈任务,以及跨圈的边界情况。
- 不适合稀疏任务:任务密集时表现优异,但当任务稀疏且跨度大时,可能浪费大量空槽。
适用场景
- 大规模定时任务,任务时间跨度大且密集。
- 时间精度要求不高,如网络超时检测、流量限速等。
形象比喻
时间轮像一个“时钟闹铃”:每个格子是一个时间点(比如 1 秒钟后或 2 分钟后),指针每到一个格子,就检查是否有任务需要执行。
· 希望各位大佬看懂了
· 那么我们下节再见吧~~~
· 毕竟不知后事如何,且听下回分解
相关文章:
Java多线程详解①①(全程干货!!!) 实现简单的线程池 || 定时器 || 简单实现定时器 || 时间轮实现定时器
这里是Themberfue 上一节讲了 线程池 线程池中的拒绝策略 等相关内容 实现简单的线程池 一个线程池最核心的方法就是 submit,通过 submit 提交 Runnable 任务来通知线程池来执行 Runnable 任务 我们简单实现一个特定线程数量的线程池,这些线程应该在…...
DAMODEL丹摩|部署FLUX.1+ComfyUI实战教程
本文仅做测评体验,非广告。 文章目录 1. FLUX.1简介2. 实战2. 1 创建资源2. 1 ComfyUI的部署操作2. 3 部署FLUX.1 3. 测试5. 释放资源4. 结语 1. FLUX.1简介 FLUX.1是由黑森林实验室(Black Forest Labs)开发的开源AI图像生成模型。它拥有12…...
请求(request)
目录 前言 request概述 request的使用 获取前端传递的数据 实例 请求转发 特点 语法 实例 实例1 实例2 【关联实例1】 域对象 组成 作用范围: 生命周期: 使用场景: 使用步骤 存储数据对象 获得数据对象 移除域中的键值…...
关于VNC连接时自动断联的问题
在服务器端打开VNC Server的选项设置对话框,点左边的“Expert”(专家),然后找到“IdleTimeout”,将数值设置为0,点OK关闭对话框。搞定。 注意,服务端有两个vnc服务,这俩都要设置ide timeout为0才行 附件是v…...
C语言strtok()函数用法详解!
strtok 是 C 标准库中的字符串分割函数,用于将一个字符串拆分成多个部分(token),以某些字符(称为分隔符)为界限。 函数原型 char *strtok(char *str, const char *delim);参数: str:…...
【docker 拉取镜像超时问题】
问题描述 在centosStream8上安装docker,使用命令sudo docker run hello-world 后出现以下错误: Error response from daemon: Get "https://registry-1.docker.io/v2/": net/http: request canceled while waiting for connection (Client.Ti…...
模拟手机办卡项目(移动大厅)--结合面向对象、JDBC、MYSQL、dao层模式,使用JAVA控制台实现
目录 1. 项目需求 2. 项目使用的技术 3.项目需求分析 3.1 实体类和接口 4.项目结构 5.业务实现 5.1 登录 5.1.1 实现步骤 5.1.2 原生代码问题 编辑 5.1.3 解决方法 1.说明: 2. ResultSetHandler结果集处理 5.1.4 代码 5.1.5 实现后的效果图 登录成功…...
机器学习—大语言模型:推动AI新时代的引擎
云边有个稻草人-CSDN博客 目录 引言 一、大语言模型的基本原理 1. 什么是大语言模型? 2. Transformer 架构 3. 模型训练 二、大语言模型的应用场景 1. 文本生成 2. 问答系统 3. 编码助手 4. 多语言翻译 三、大语言模型的最新进展 1. GPT-4 2. 开源模型 …...
C++:探索哈希表秘密之哈希桶实现哈希
文章目录 前言一、链地址法概念二、哈希表扩容三、哈希桶插入逻辑四、析构函数五、删除逻辑六、查找七、链地址法代码实现总结 前言 前面我们用开放定址法代码实现了哈希表: C:揭秘哈希:提升查找效率的终极技巧_1 对于开放定址法来说&#…...
具身智能高校实训解决方案——从AI大模型+机器人到通用具身智能
一、 行业背景 在具身智能的发展历程中,AI 大模型的出现成为了关键的推动力量。这些大模型具有海量的参数和强大的语言理解、知识表示能力,能够为机器人的行为决策提供更丰富的信息和更智能的指导。然而,单纯的大模型在面对复杂多变的现实…...
【消息序列】详解(8):探秘物联网中设备广播服务
目录 一、概述 1.1. 定义与特点 1.2. 工作原理 1.3. 应用场景 1.4. 技术优势 二、截断寻呼(Truncated Page)流程 2.1. 截断寻呼的流程 2.2. 示例代码 2.3. 注意事项 三、无连接外围广播过程 3.1. 设备 A 启动无连接外围设备广播 3.2. 示例代…...
【RL Base】强化学习核心算法:深度Q网络(DQN)算法
📢本篇文章是博主强化学习(RL)领域学习时,用于个人学习、研究或者欣赏使用,并基于博主对相关等领域的一些理解而记录的学习摘录和笔记,若有不当和侵权之处,指出后将会立即改正,还望谅…...
深入浅出 Python 网络爬虫:从零开始构建你的数据采集工具
在大数据时代,网络爬虫作为一种数据采集技术,已经成为开发者和数据分析师不可或缺的工具。Python 凭借其强大的生态和简单易用的语言特点,在爬虫领域大放异彩。本文将带你从零开始,逐步构建一个 Python 网络爬虫,解决实…...
美国发布《联邦风险和授权管理计划 (FedRAMP) 路线图 (2024-2025)》
文章目录 前言一、战略目标实施背景2010年12月,《改革联邦信息技术管理的25点实施计划》2011年2月,《联邦云计算战略》2011年12月,《关于“云计算环境中的信息系统安全授权”的首席信息官备忘录》2022年12月,《FedRAMP 授权法案》…...
Python语法基础(三)
🌈个人主页:羽晨同学 💫个人格言:“成为自己未来的主人~” 我们这篇文章来说一下函数的返回值和匿名函数 函数的返回值 我们先来看下面的这一段函数的定义代码 # 1、返回值的意义 def func1():print(111111111------start)num166print…...
云计算之elastaicsearch logstach kibana面试题
1.ELK是什么? ELK 其实并不是一款软件,而是一整套解决方案,是三个软件产品的首字母缩写 Elasticsearch:负责日志检索和储存 Logstash:负责日志的收集和分析、处理 Kibana:负责日志的可视化 这三款软件都是开源软件,通常是配合使用,而且又先后归于 Elastic.co 公司名下,…...
【已解决】git push需要输入用户名和密码问题
解决方法: 1)查看使用的clone方式: git remote -v 2)若为HTTPS,删除原clone方式: git remote rm origin 3)添加新的clone方式: git remote add origin gitgithub.com:zludon/git_test.git …...
python的字符串处理
需求: 编写一个程序,输入一段英文句子,统计每个单词的长度,并将单词按照长度从短到长排序。 程序逻辑框图 1、用户输入一句英文句子。 2、对输入的句子进行预处理(去空格并分割为单词列表)。 3、统计每个单…...
【线程】Java多线程代码案例(2)
【线程】Java多线程代码案例(2) 一、定时器的实现1.1Java标准库定时器1.2 定时器的实现 二、线程池的实现2.1 线程池2.2 Java标准库中的线程池2.3 线程池的实现 一、定时器的实现 1.1Java标准库定时器 import java.util.Timer; import java.util.Timer…...
虚拟机之间复制文件
在防火墙关闭的前提下,您可以通过几种不同的方法将文件从一个虚拟机复制到另一个虚拟机。这里,我们假设您想要从 IP 地址为 192.168.4.5 的虚拟机上的 /tmp 文件夹复制文件到当前虚拟机(192.168.4.6)的 /tmp 文件夹下。以下是几种…...
XCTF-web-easyupload
试了试php,php7,pht,phtml等,都没有用 尝试.user.ini 抓包修改将.user.ini修改为jpg图片 在上传一个123.jpg 用蚁剑连接,得到flag...
Qt/C++开发监控GB28181系统/取流协议/同时支持udp/tcp被动/tcp主动
一、前言说明 在2011版本的gb28181协议中,拉取视频流只要求udp方式,从2016开始要求新增支持tcp被动和tcp主动两种方式,udp理论上会丢包的,所以实际使用过程可能会出现画面花屏的情况,而tcp肯定不丢包,起码…...
前端倒计时误差!
提示:记录工作中遇到的需求及解决办法 文章目录 前言一、误差从何而来?二、五大解决方案1. 动态校准法(基础版)2. Web Worker 计时3. 服务器时间同步4. Performance API 高精度计时5. 页面可见性API优化三、生产环境最佳实践四、终极解决方案架构前言 前几天听说公司某个项…...
遍历 Map 类型集合的方法汇总
1 方法一 先用方法 keySet() 获取集合中的所有键。再通过 gey(key) 方法用对应键获取值 import java.util.HashMap; import java.util.Set;public class Test {public static void main(String[] args) {HashMap hashMap new HashMap();hashMap.put("语文",99);has…...
1688商品列表API与其他数据源的对接思路
将1688商品列表API与其他数据源对接时,需结合业务场景设计数据流转链路,重点关注数据格式兼容性、接口调用频率控制及数据一致性维护。以下是具体对接思路及关键技术点: 一、核心对接场景与目标 商品数据同步 场景:将1688商品信息…...
家政维修平台实战20:权限设计
目录 1 获取工人信息2 搭建工人入口3 权限判断总结 目前我们已经搭建好了基础的用户体系,主要是分成几个表,用户表我们是记录用户的基础信息,包括手机、昵称、头像。而工人和员工各有各的表。那么就有一个问题,不同的角色…...
MVC 数据库
MVC 数据库 引言 在软件开发领域,Model-View-Controller(MVC)是一种流行的软件架构模式,它将应用程序分为三个核心组件:模型(Model)、视图(View)和控制器(Controller)。这种模式有助于提高代码的可维护性和可扩展性。本文将深入探讨MVC架构与数据库之间的关系,以…...
镜像里切换为普通用户
如果你登录远程虚拟机默认就是 root 用户,但你不希望用 root 权限运行 ns-3(这是对的,ns3 工具会拒绝 root),你可以按以下方法创建一个 非 root 用户账号 并切换到它运行 ns-3。 一次性解决方案:创建非 roo…...
爬虫基础学习day2
# 爬虫设计领域 工商:企查查、天眼查短视频:抖音、快手、西瓜 ---> 飞瓜电商:京东、淘宝、聚美优品、亚马逊 ---> 分析店铺经营决策标题、排名航空:抓取所有航空公司价格 ---> 去哪儿自媒体:采集自媒体数据进…...
什么?连接服务器也能可视化显示界面?:基于X11 Forwarding + CentOS + MobaXterm实战指南
文章目录 什么是X11?环境准备实战步骤1️⃣ 服务器端配置(CentOS)2️⃣ 客户端配置(MobaXterm)3️⃣ 验证X11 Forwarding4️⃣ 运行自定义GUI程序(Python示例)5️⃣ 成功效果
,这里的最初版本其实为: