当前位置: 首页 > news >正文

JavaEE 初阶(13)——多线程11之“定时器”

目录

一. 什么是“定时器”

二. 标准库的定时器 

三. 定时器的实现 MyTimer

3.1 分析思路

1. 创建执行任务的类。

 2. 管理任务

3. 执行任务

3.2 线程安全问题

四. 拓展


一. 什么是“定时器”

   定时器是软件开发中的一个重要组件,类似于一个“闹钟”,达到一个设定的时间之后,就执行某个指定好的代码。

   定时器是一种实际开发中非常常用的组件:比如网络通信中,如果对方 500ms 内没有返回数据,则断开连接尝试重连;比如一个 Map,希望里面的某个 key 在 3s 之后过期(自动删除)
   类似于这样的场景就需要用到定时器。

二. 标准库的定时器 

   Timer类是一个线程安全的工具类,用于在指定的时间后执行或定期执行任务。它基于绝对时间,而不是基于固定时间间隔调度任务。Timer类比较简单,适用于简单的定时任务,但不适合处理复杂的调度需求。(ScheduleExecutorService 提供了更灵活的线程管理功能,适用于复杂的调度场景,并且能够处理异常)

  • 标准库中(java.util.Timer ),Timer 类的核心方法为 schedule
  • schedule 包含两个参数,第一个参数指定即将要执行的任务代码,第二个参数指定多长时间之后执行(单位为毫秒)
import java.util.Timer;
import java.util.TimerTask;public class Timer1 {public static void main(String[] args) throws InterruptedException {Timer timer1 = new Timer();timer1.schedule(new TimerTask() {@Overridepublic void run() {System.out.println("hello delay"+" " +1000);}},1000);timer1.schedule(new TimerTask() {@Overridepublic void run() {System.out.println("hello delay"+" " +2000);}},2000);timer1.schedule(new TimerTask() {@Overridepublic void run() {System.out.println("hello delay"+" " +3000);}},3000);// 等待提交的任务执行完毕后,调用 cancel 便可结束进程Thread.sleep(4000);timer1.cancel();}
}

三. 定时器的实现 MyTimer

3.1 分析思路

对于定时器来说:

  1. 创建描述一个要执行的任务(任务内容 + 执行任务时间)的类
  2. 管理多个任务,通过一定的数据结构,把多个任务存起来
  3. 有专门的线程,执行这里的任务
1. 创建执行任务的类。

   我们在调用 schedule 时,传的是延迟时间 “delay” 值。但是,描述任务时,不太建议使用 delay

表示,最好使用 “绝对时间”(时间戳)来表示~~

public class MyTimerTask implements Comparable<MyTimerTask>{//此处这里的 time,通过毫秒时间戳,表示这个任务具体啥时候执行private long time;private Runnable runnable;public  MyTimerTask(Runnable runnable,long delay){this.time = System.currentTimeMillis() + delay;this.runnable = runnable;}public void run(){runnable.run();}public long getTime(){return time;}@Overridepublic int compareTo(MyTimerTask o) {//比如,当前时间是 10:30,任务时间是 12:00,不应该执行//如果当前时间是 10:30,任务时间是 10:29,应该执行//谁减去谁,可以通过实验判断return (int) (this.time - o.time);}
}
 2. 管理任务

   使用 List 管理任务,不是一个好选择——因为后续执行列表中的任务时,就需要依次遍历每个元素;执行完毕后,还需要把对应的任务从 List 中删除掉。

   我们需要按照时间来执行这里的任务。只要能够确定所有任务中,时间最早的任务,判定它是否到该执行的时间即可。如果时间最早的任务还没到执行时间,其他任务更不可能到时间了。因此,我们使用堆数据结构(涉及到队列中的元素排序时,考虑堆)——PriorityQueue<MyTimerTask>(优先级队列管理元素时,需要有比较方法,才能排序存储。因此,在实现 MyTimerTask 类时,要继承 Comparable<MyTimerTask> 接口,重写 compareTo比较方法)

3. 执行任务

   当创建 MyTimer 对象,调用无参构造方法时,便创建一个线程,循环执行从队列中取出任务的操作:取出队列中 “绝对时间” 最早的任务——如果当前时间 >= 此任务的时间(已经到达此任务的执行时间),便可调用run方法执行,执行完毕后从队列中删除; 如果当前时间 < 此任务的时间(没到此任务的执行时间),则继续执行循环。(所以,在实现 MyTimerTask 类时,要有 run方法 和 getTime方法)

   除此之外 ,还需要有 schedule 方法添加任务。

import java.util.PriorityQueue;public class MyTimer {private final PriorityQueue<MyTimerTask> queue = new PriorityQueue<>();public MyTimer() {Thread t = new Thread(() -> {while (true){if(queue.isEmpty()){ continue;}MyTimerTask task = queue.peek();//判断是否满足执行条件if (System.currentTimeMillis() >= task.getTime()) {task.run();//执行完后,便从队列中删除queue.poll();}else{continue;}}});t.start();}public void schedule(Runnable runnable, long delay) {MyTimerTask task = new MyTimerTask(runnable, delay);queue.offer(task);}
}
3.2 线程安全问题

   当前这个代码,是没有考虑线程安全问题的。

   PriorityQueue 这个类自身,是非线程安全的,并且又是多个线程来进行操作,一定存在线程安全问题的风险。因此,要在涉及队列相关操作的地方加锁

import java.util.PriorityQueue;public class MyTimer {private final PriorityQueue<MyTimerTask> queue = new PriorityQueue<>();private final Object locker = new Object();public MyTimer() {Thread t = new Thread(() -> {while (true){synchronized(locker){if(queue.isEmpty()){ continue;}MyTimerTask task = queue.peek();if (System.currentTimeMillis() >= task.getTime()) {task.run();queue.poll();}else{continue;}}}});t.start();}public void schedule(Runnable runnable, long delay) {synchronized(locker){MyTimerTask task = new MyTimerTask(runnable, delay);queue.offer(task);}}
}

   但是,加完锁以后,又出现了线程安全问题。 

1)初始情况下,如果队列中,没有任何元素。此时,就会在短时间内执行大量循环,这样的执行是没有意义的,导致“线程饿死”。

   因此,我们需要添加 wait 和 notify 机制:队列为空时,进行等待;添加任务时,就唤醒线程。


 2)假设队列中,已经包含元素了,并且当前时间是 10:45,任务时间 12:00(类似于,我定了12:00的闹钟,现在是 10:45)。在判断任务是否满足执行条件时,不满足就会一直循环(相当于每隔一会儿就看一眼闹钟),这样无意义的执行就一直占用着cpu资源,导致 “线程饿死”。

   因此,我们需要添加一个有等待期限的 wait(等待 1h15min 就会执行),当到达任务执行时间,wait 就结束了。如果在等待过程中,又再次调用 schedule 方法,也会唤醒这里的 wait,进行新一轮的判断。


 线程安全版:

import java.util.PriorityQueue;public class MyTimer {private final PriorityQueue<MyTimerTask> queue = new PriorityQueue<>();private final Object locker = new Object();public MyTimer() {Thread t = new Thread(() -> {try {while (true) {synchronized (locker) {while (queue.isEmpty()) {//如果还没添加任务,会不断循环执行判断,出现线程饿死。//continue;//因此,使用wait等待,当添加任务后唤醒locker.wait();}MyTimerTask task = queue.peek();if (System.currentTimeMillis() >= task.getTime()) {task.run();queue.poll();} else {//如果还没到任务执行时间,依旧不断循环判断,出现线程饿死。//continue;//因此,使用有等待期限的 wait,计算执行的时间与当前时间的差值//当添加新的任务后,wait 被唤醒,再进行新的判断locker.wait(task.getTime() - System.currentTimeMillis());}}}} catch (InterruptedException e) {e.printStackTrace();}});t.start();}public void schedule(Runnable runnable, long delay) {synchronized (locker){MyTimerTask task = new MyTimerTask(runnable, delay);queue.offer(task);// 唤醒 waitlocker.notify();}}
}

 * 能否将第二处的 wait 改为 sleep 呢? ——不能!!

 不应该使用sleep,可能存在以下情况:

     1)在 sleep 阻塞1h15min 的过程中,新来了一个时间更早的任务,比如 11:30 要执行。如果使用 wait ,每次新来的任务,都会把 wait 唤醒,重新设定 wait 的等待时间。而 sleep 不会被唤醒,依旧在阻塞着.....

     2)sleep 休眠的时候,不会释放锁。因此,在休眠的时候就是“抱着锁”,其他人想拿锁就拿不到了。


 * PriorityQueue 是线程不安全的类,能否使用 PriorityBlockingQueue 线程安全的阻塞队列呢?——不能!! 

   如果使用线程安全的队列,会导致代码中从 一把锁 变成 两把锁,很容易出现死锁的情况,比如持有和请求的情况(并非100%一定出现,但是需要程序员精心控制加锁顺序,使得编写代码的复杂度提高了。如果通篇代码 只有一把锁,就能更容易地解决问题)


 * 如果某个任务的执行时间过长怎么办? 

   此处只有一个线程,串行执行每一个任务。如果某个任务执行时间过长,可能会影响到定时器的准确性(特别是对于周期性任务),使后续的任务被延迟——解决方案,多创建几个这样的工作线程。

四. 拓展

   业界实现定时器,除了基于优先级队列的方式之外,还有一种经典的实现方式——“时间轮”

   “时间轮” 是一种设计巧妙的数据结构,通常用于高效处理大量的定时任务。它是定时器任务调度的一种优化方案,尤其在需要调度大量短任务的场景中非常有效。时间轮的概念类似于钟表,它将时间划分为固定数量的槽位,每个槽位对应一个时间间隔,用于存储该时间间隔内需要执行的任务。

   (这只是简单介绍,不做过多讨论,可自行查阅~~)

   定时器特别重要,也特别常用,尤其是后端开发中,和 “阻塞队列” 类似,也会有专门服务器,用来在分布式系统中实现定时器这样的效果。

  • hashmap --> redis
  • 阻塞队列 --> 消息队列
  • 定时器 --> 定时任务

相关文章:

JavaEE 初阶(13)——多线程11之“定时器”

目录 一. 什么是“定时器” 二. 标准库的定时器 三. 定时器的实现 MyTimer 3.1 分析思路 1. 创建执行任务的类。 2. 管理任务 3. 执行任务 3.2 线程安全问题 四. 拓展 一. 什么是“定时器” 定时器是软件开发中的一个重要组件&#xff0c;类似于一个“闹钟”&#xff0…...

2024最新全开源付费进群系统源码二开修复版 支持易支付

内容目录 一、详细介绍二、效果展示1.部分代码2.效果图展示 三、学习资料下载 一、详细介绍 全开源付费进群系统源码&#xff0c;开源无加密无授权&#xff0c;优化电脑端访问布局&#xff0c;支持dai理&#xff0c;对接易支付通道&#xff0c;dai理可以配置自己易支付接口&am…...

【奥顺苹果CMS二开泛目录4.X版】PHP站群程序新增首页堆砌关键词新增四套seo模板

演示站&#xff08;赠送四套模板&#xff09;&#xff1a; https://macfan.qdwantong.com https://macfan2.qdwantong.com https://macfan3.qdwantong.com https://macfan4.qdwantong.com 4.X版程序特色功能&#xff1a; 后台除了可以设置干扰码、转码、插入符号和拼音这…...

day06 项目实践:router,axios

vue组件的生命周期钩子 今天几乎没有讲什么新内容&#xff0c;就是一起做项目&#xff0c;只有一个小小的知识点&#xff0c;就是关于vue组件的生命周期钩子&#xff0c;其中最重要的四个函数—— beforeCreate()&#xff1a;组件创建之间执行 created()&#xff1a;组件创建…...

⌈ 传知代码 ⌋ 基于矩阵乘积态的生成模型

&#x1f49b;前情提要&#x1f49b; 本文是传知代码平台中的相关前沿知识与技术的分享~ 接下来我们即将进入一个全新的空间&#xff0c;对技术有一个全新的视角~ 本文所涉及所有资源均在传知代码平台可获取 以下的内容一定会让你对AI 赋能时代有一个颠覆性的认识哦&#x…...

软件测试必备技能

在软件测试领域&#xff0c;以下是一些必备的技能和能力&#xff0c;可以帮助你成为一名优秀的软件测试工程师&#xff1a; 1. 测试基础知识&#xff1a; 熟悉软件测试的基本概念、原则和流程&#xff0c;包括不同类型的测试&#xff08;如单元测试、集成测试、系统测试&#…...

TL3568编译uboot报错

编译uboot前&#xff0c;需要 ① sudo apt-get install device-tree-compiler 否则会报“ERROR: No dtc” ② sudo apt install python 装个Python2&#xff0c;否则会报“ERROR: No python2”...

qiankun 微前端 隔离子应用样式,解决 ant-design-vue 子应用样式污染问题(已落地)

样式冲突产生原因 先分析乾坤qiankun 构建之后&#xff0c;会根据你的配置 给每个子应用生成一个id&#xff0c; 当加载到对应子应用的时候&#xff0c;就把内容放到对应的id 标签里去&#xff0c; 这样能有效的隔离 js 代码&#xff0c;但是样式是加载在全局的 所以 当两个子…...

一个前后端分离架构的低代码开发平台,支持微服务架构,支持开发SAAS项目(附源码)

前言 在当前的企业软件开发领域&#xff0c;开发者常常面临着代码重复性高、开发效率低、项目周期长等挑战。现有的软件解-决方案往往难以满足快速变化的市场需求&#xff0c;特别是在SAAS项目、企业信息管理系统&#xff08;MIS&#xff09;、内部办公系统&#xff08;OA&…...

whisper+whisperx ASR加对齐

忘了怎么安装了&#xff0c;这里记录一下整理出来的类&#xff0c;不过这个 from chj.comm.pic import *import json import whisper import whisperx import gcclass Warp_whisper:def __init__(self, language"zh", device"cuda", compute_type"fl…...

【已解决】YOLOv8加载模型报错:super().__init__(torch._C.PyTorchFileReader(name_or_buffer))

《博主简介》 小伙伴们好&#xff0c;我是阿旭。专注于人工智能、AIGC、python、计算机视觉相关分享研究。 ✌更多学习资源&#xff0c;可关注公-仲-hao:【阿旭算法与机器学习】&#xff0c;共同学习交流~ &#x1f44d;感谢小伙伴们点赞、关注&#xff01; 《------往期经典推…...

中国象棋 纯网页前端 演示与下载

https://andi.cn/app/chess/...

学习大数据DAY29 python基础语法2

目录 调试---debug tuple&#xff08;元组&#xff09; set&#xff08;集合&#xff09; dict&#xff08;字典&#xff09; 转换 推导式 上机练习 3 函数 参数 不定长参数 值传递与引用传递 局部和全局变量 上机练习 4 调试---debug 1. 先设置断点 2. 点击调试…...

自动化测试常用函数(Java方向)

目录 一、元素的定位 1.1 cssSelector 1.2 xpath 1.2.1 获取HTML页面所有的节点 1.2.2 获取HTML页面指定的节点 1.2.3 获取⼀个节点中的直接子节点 1.2.4 获取⼀个节点的父节点 1.2.5 实现节点属性的匹配 1.2.6 使用指定索引的方式获取对应的节点内容 二、操作测试对…...

申瓯通信设备有限公司在线录音管理系统(复现过程)

漏洞简介 申瓯通信设备有限公司在线录音管理系统 index.php接口处存在任意文件读取漏洞&#xff0c;恶意攻击者可能利用该漏洞读取服务器上的敏感文件&#xff0c;例如客户记录、财务数据或源代码&#xff0c;导致数据泄露 一.复现过程 fofa搜索语句:title"在线录音管…...

【C++进阶学习】第十一弹——C++11(上)——右值引用和移动语义

前言&#xff1a; 前面我们已经将C的重点语法讲的大差不差了&#xff0c;但是在C11版本之后&#xff0c;又出来了很多新的语法&#xff0c;其中有一些作用还是非常大的&#xff0c;今天我们就先来学习其中一个很重要的点——右值引用以及它所扩展的移动定义 目录 一、左值引用和…...

JavaScript 监听 localStorage 的变化

使用 JavaScript 监听 localStorage 的变化 在Web开发中,localStorage是一种非常常用的本地存储机制。它允许我们在浏览器中存储键值对数据,即使用户关闭了浏览器或刷新页面,数据也不会丢失。但是,有时我们需要实时监控 localStorage 的变化,以便能够及时做出响应。在本文中,我…...

Java 中 HashMap 和 Hashtable 的联系

目录 相同 不同 1. 继承的父类不同 2. 线程安全性不同 3. 包含的 contains 方法不同 4. toString方法不同 5. 是否允许null值不同 6. 计算hash值的方式不同 7. 计算索引位置的方法不同 8. 初始化容量不同 9. 扩容方式不同 10. 内部存储策略不同&#xff08;此处讨论…...

Web3 开发教程

引言 Web3 是指第三代互联网&#xff0c;其核心特征之一是去中心化。通过区块链技术和智能合约&#xff0c;Web3 应用程序&#xff08;dApps&#xff09;能够在无需中心化服务器的情况下运行。本文将引导你完成一个简单的 Web3 应用程序的开发过程&#xff0c;包括环境搭建、智…...

傻瓜式PHP-Webshell免杀学习手册,零基础小白也能看懂

项目描述 一、PHP相关资料 PHP官方手册&#xff1a; https://www.php.net/manual/zh/ PHP函数参考&#xff1a; https://www.php.net/manual/zh/funcref.php 菜鸟教程&#xff1a; https://www.runoob.com/php/php-tutorial.html w3school&#xff1a; https://www.w3school…...

Banana Vision Studio在汽车设计中的曲面分析应用

Banana Vision Studio在汽车设计中的曲面分析应用 1. 引言 在汽车设计领域&#xff0c;曲面质量直接决定了一款车的视觉美感和空气动力学性能。传统的曲面分析方法往往需要设计师手动检查每个曲面的连续性、曲率变化和光顺度&#xff0c;这个过程既耗时又容易出错。现在&…...

Claude Code风格与LiuJuan20260223Zimage代码生成的对比

Claude Code风格与LiuJuan20260223Zimage代码生成的对比 在代码生成领域&#xff0c;不同的模型展现出各自独特的风格和特点。今天我们来对比分析Claude Code与LiuJuan20260223Zimage这两个模型在代码生成方面的表现差异&#xff0c;通过实际案例展示它们各自的优势和适用场景…...

计算机毕业设计springboot基于Vue.js的企业资产管理系统 基于SpringBoot与Vue.js的企业固定资产全生命周期管理平台 采用前后端分离架构的企业设备资产数字化运营系统

计算机毕业设计springboot基于Vue.js的企业资产管理系统&#xff08;配套有源码 程序 mysql数据库 论文&#xff09; 本套源码可以在文本联xi,先看具体系统功能演示视频领取&#xff0c;可分享源码参考。随着企业规模的扩张与业务复杂度的提升&#xff0c;传统手工记录模式已难…...

在技术债沼泽建城堡:让裁员成本高到绝望

在软件测试领域&#xff0c;技术债&#xff08;Technical Debt&#xff09;常被视为隐藏的沼泽——它悄无声息地累积&#xff0c;拖慢开发效率&#xff0c;增加系统风险。然而&#xff0c;聪明的测试从业者能将这片沼泽转化为坚固城堡&#xff0c;让裁员成本高到企业望而却步。…...

智能搜索新选择:通义千问3-VL-Reranker-8B部署与API调用教程

智能搜索新选择&#xff1a;通义千问3-VL-Reranker-8B部署与API调用教程 你是否曾为搜索结果不够精准而烦恼&#xff1f;比如在电商平台找“适合户外运动的红色背包”&#xff0c;结果搜出来的商品图片里背包却是蓝色的。或者想找一段“海边日落”的视频素材&#xff0c;系统却…...

PP-DocLayoutV3模型部署详解:从Docker镜像到RESTful API服务

PP-DocLayoutV3模型部署详解&#xff1a;从Docker镜像到RESTful API服务 你是不是拿到一个封装好的AI模型Docker镜像&#xff0c;却不知道如何把它变成一个对外提供服务的API&#xff1f;或者觉得官方文档只讲了怎么跑起来&#xff0c;但离真正的生产级服务还差那么几步&#…...

Z-Image-Turbo-辉夜巫女部署案例:单卡3090高效运行LoRA定制文生图模型

Z-Image-Turbo-辉夜巫女部署案例&#xff1a;单卡3090高效运行LoRA定制文生图模型 1. 项目概述 Z-Image-Turbo-辉夜巫女是一款基于LoRA技术优化的文生图模型&#xff0c;专门用于生成高质量的辉夜巫女风格图片。该模型在单卡NVIDIA 3090上实现了高效运行&#xff0c;通过Xinf…...

企业 AI 智能体:从 Demo 到规模化落地的技术架构与工程实践

在大模型技术普及的今天&#xff0c;几乎所有企业的技术团队都尝试过搭建 AI 智能体&#xff1a;从简单的客服问答机器人&#xff0c;到能处理基础业务的 AI 助手&#xff0c;大多都能快速做出一个效果亮眼的 Demo。但现实是&#xff0c;超过 90% 的 AI 智能体&#xff0c;最终…...

单北斗GNSS形变监测是什么?主要有如何应用于大坝监测?

单北斗GNSS形变监测是一种利用卫星技术进行位移监测的高精度系统&#xff0c;广泛应用于大坝、桥梁等基础设施的安全监测。该系统通过接收GPS信号&#xff0c;能够实时获取目标点的三维位置变化&#xff0c;提供可靠的数据支持。在应用过程中&#xff0c;用户可以根据具体监测需…...

基于ESP32的智能猫用饮水器设计与实现

1. 项目概述“猫猫喂水器”是一个面向家庭宠物场景的嵌入式智能饮水管理终端&#xff0c;核心目标是解决用户短期离家期间猫咪饮水保障问题。系统通过非接触式水位监测、闭环控制逻辑与远程交互能力&#xff0c;实现“无人值守下的按需补水”。其设计并非追求高精度工业级液位计…...