【JavaEE初阶 — 多线程】定时器的应用及模拟实现

     c96f743646e841f8bb30b2d242197f2f.gif

ddb5ae16fc92401ea95b48766cb03d96.jpeg692a78aa0ec843629a817408c97a8b84.gif

目录

    1. 标准库中的定时器    

     1.1 Timer 的定义    

     1.2 Timer 的原理     

     1.3 Timer 的使用    

    1.4 Timer 的弊端    

     1.5 ScheduledExecutorService     

    2. 模拟实现定时器    

    2.1 实现定时器的步骤     

    2.1.1  定义类描述任务    

     定义类描述任务    

     第一种定义方法     

    第二种定义方法    

     定义 MyTimerTask 类的时间属性    

     完善描述任务类的比较规则    

     2.1.2 使用优先级队列实现 Timer    

     2.1.3  实现 schedule() 方法    

     对该方法的简单说明    

    把任务添加到队列中    

     2.1.4 扫描线程中负责执行队列中的任务    

     创建 worker    

     模拟队列为空时的阻塞效果    

     完善对任务的时间属性进行比较的操作     

     处理线程安全问题     

     安排任务    

     程序运行结果    

     2.2 忙等问题    

     2.3 模拟实现定时器完整代码    

     3.  拓展时间轮     

     4. 优先级队列 & 时间轮的优缺点    


 


    1. 标准库中的定时器    


     1.1 Timer 的定义    


  • Java 的Timer是一个用于调度任务的工具类,用于在未来某个时刻执行任务或周期性地执行任务。
  • Timer类一般与 TimerTask 搭配使用,其中 TimerTask 是一个需要执行的任务。
  • 适用于简单的定时任务,如定时更新、定期发送报告等。

     1.2 Timer 的原理     


Timer可以实现延时任务,也可以实现周期性任务,它的核心就是一个优先队列和封装的执行任务的线程。


    实现原理    


  • 维持一个小顶堆,即最快需要执行的任务排在优先队列的第一个,根据堆的特性我们知道插入和删除的时间复杂度都是O(log n)。
  • 然后有个TimerThread线程,不断地拿排着的 第一个任务 的执行时间,和当前时间做对比。
  • 如果时间到了先看看这个任务是不是周期性执行的任务:
  • 如果是周期性任务,则修改当前任务时间为下次执行的时间;
  • 如果不是周期性任务,则将任务从优先队列中移除,最后执行任务。
  • 如果时间还未到则调用wait() 等待。

 

可以看出Timer,实际就是根据任务的执行时间,维护了一个优先队列,并且起了一个线程,不断地拉取任务执行。


     1.3 Timer 的使用    


编译器一般都是Runnable 来描述任务,但是在定时器这里稍微特殊一点,把 Runnable 封装成了  TimerTask;TimeTask 本质上是一个抽象类:


    示例一     

    程序运行结果    

进程一直没有结束,是因为 Timer 和线程池一样 ,都包含了前台线程,阻止进程结束;


    1.4 Timer 的弊端    


  • 首先优先队列的插入和删除的时间复杂度是O(logn),当数据量大的时候,频繁的入堆出堆性能有待考虑。
  • 并且是单线程执行,那么如果一个任务执行的时间过久,则会影响下一个任务的执行时间(当然我们设置任务的 run(),要是异步执行也行)。
  • 并且从它对异常没有做什么处理,所以一个任务出错的时候会导致之后的任务都无法执行。

     1.5 ScheduledExecutorService     


  • ScheduledExecutorService 是 Java5 引入的替代方案,功能更强大。
  • 它支持多线程并行调度任务,能更好地处理任务调度的复杂场景
  • 因为使用线程池进行任务调度,所以不会因某个任务的异常终止,而导致其他任务停止。

  • 并且它提供了更灵活的API,可以更精细地控制任务的执行周期和策略。
  • 推荐使用ScheduledExecutorService 替代Timer。

    示例二    

     注意事项     


    程序运行结果    


    2. 模拟实现定时器    


    2.1 实现定时器的步骤     


    2.1.1  定义类描述任务    


     定义类描述任务    

通过 MyTimerTask类 来描述,在定时器定时的时间结束后,线程要执行的任务 :


     第一种定义方法     

基于抽象类的方式定义MyTimerTask,并实现 Runnable 接口,并且重写 run();

这样的定义虽然确实可以,写起来有点麻烦,还有另外的写法;


    第二种定义方法    

可以不把 MyTimerTask类 设置成抽象类,而是在MyTimerTask类的成员中,持有一个 Runnable:

后续通过构造方法参数,把定义的任务传进来:

上述两种写法都是可以的;  


     定义 MyTimerTask 类的时间属性    

MyTimerTask 不但要描述要执行的任务,还要记录什么时候任务被执行:

并提供相应的方法来获取任务执行时间,和任务执行方法:


     完善描述任务类的比较规则    

  • 因为要通过优先级队列比较任务的执行时间,对于要比较的元素是 int,String 这种本身就有明确比较规则的对象,可以不额外指定;
  • 但是我们自己定义的类 MyTimerTask 是没有明确比较规则的,所以我们需要给  MyTimerTask 实现比较规则,实现 Comparable 接口,重写 compareTo() 方法;

  • 这个优先级队列是小根堆,让时间少的任务先执行,如果分不清楚建立的是小根堆还是大根堆,就排列组合,直到找到合适的计算表达式;
  • time 之间的计算结果是 long 类型的,compareTo() (是优先级队列内部调用的),返回类型是 int,所以需要强转;

     2.1.2 使用优先级队列实现 Timer    



  • 定时器的构成是一个优先级队列(不要使用PriorityBlockingQueue,容易死锁);
  • 队列中的每个元素是一个 Task 对象,因此我们自己实现 Timer,可以把泛型参数设置成刚刚实现的,用来描述任务的类 MyTimerTask:


     2.1.3  实现 schedule() 方法    


     对该方法的简单说明    


    把任务添加到队列中    


     2.1.4 扫描线程中负责执行队列中的任务    


     创建 worker    

同时有一个worker线程一直扫描队首元素,看队首元素是否需要执行 ;所以通过 MyTimer 的构造方法,来创建 Thread 对象:

task 中带有一个时间属性,队首元素就是即将要执行的任务。


     模拟队列为空时的阻塞效果    

如果在往队列中取任务的时候,发现队列为空,则模拟出线程被队列阻塞的效果:

只有当任务执行完才可以将其移除出优先级队列,否则只能通过 peek() 取任务;


     完善对任务的时间属性进行比较的操作     

和线程池不同,线程池是只要队列不为空,就立即取任务并执行; 

但是 worker 需要关注队首元素的时间属性,系统时间到了任务执行时间,队首元素才会被取出并且执行,否则时间不到,任务不能执行:

在拿到任务后,我们需要比较当前系统时间是否已经到任务执行时间,没有到就继续通过 continue 的方式模拟阻塞 ,否则执行该任务,并且在执行完后出队列。


     处理线程安全问题     

当前调用 schedule 是一个线程,定时器内部又有一个线程,多个线程操作同一个队列,一定涉及到线程安全问题,所以我们要给 submit()  和 worker 中的操作加锁:

构造方法本身可以写synchronized,但是这个地方不能这么写,要保护的逻辑是 lambda表达式中的 run() ,run() 和构造方法是两个不同的方法;


     安排任务    


     程序运行结果    

  • worker 需要关注队首元素的时间属性,系统时间到了任务执行时间,队首元素才会被取出并且执行,否则时间不到,任务不能执行;
  • 和线程池不同,线程池是只要队列不为空,就立即取任务并执行; 
  • worker 在发现队列为空时,会陷入 continue 模拟出来的阻塞等待,进程继续保持运行状态;

     2.2 忙等问题    


    什么是忙等    


  • 定时器在执行上述任务,会出现忙等问题,忙等并没有实质性地做任何工作,只是在等待。
  • 但是忙等又和 sleep 这样的等待不同:
  • sleep 这样的等待是会释放 CPU 资源的等待,如果这个线程什么都不干,就不参与CPU资源的调度,把CPU资源让给其他线程;
  • 但是忙等 既要消耗 CPU 资源,又不执行任务,这样的设定是不科学的,所以我们就需要针对这样的忙等,进行进一步的优化

    出现忙等问题的代码   



    通过 wait() ,notify() 解决忙等问题    


当第一个 wait() 被触发,说明队列为空,只需要调用 schedule() 往队列中添加任务即可;

schedule() 里的操作恰好也需要锁,所以添加wait() notify() (都需要搭配锁来使用)来减轻程序忙等的现象,顺理成章;

因此唤醒第一个 wait() 的 notify() 在 schedule() 中设置;

    对于解决两处忙等问题的思路如下    


  • 第一个wait(),一被唤醒了,下面的逻辑被执行,是和当前 wait() 的判断条件(队列是否为空),是不冲突的;
  • 队列为空,只要时间到了,也能执行下面的逻辑,因此需要把 if() 改成 while(),让第一个 wait() 被唤醒后,再次判断是否满足唤醒条件;


  • 第二个 wait() 的唤醒条件,只看时间是否到了,到了才能执行下面的逻辑;
  • 所以第二个wait被打断了,else也会判断当前系统时间,是否已经到任务执行时间,因此不需要把 if() 改成while 

    补充    


单击其中一个 locker,其他 locker 同时显示,证明获取的 locker 为同一个;

虽然是在 lamda 中的对 this 加锁,看着像是 this 指向匿名内部类,然后对匿名内部类加锁,但是实际上是 lamda 的变量捕获,使得此处的this指向 MyTimer 这个外部类;


     2.3 模拟实现定时器完整代码    


package Thread;import java.util.PriorityQueue;
import java.util.TimerTask;
import java.util.concurrent.Executors;class MyTimerTask implements Comparable<MyTimerTask>{private Runnable task;//记录要执行的任务的时刻private long time;public MyTimerTask(Runnable task, long time) {this.task = task;this.time = time;}@Overridepublic int compareTo(MyTimerTask o) {return (int)(this.time-o.time);}public long getTime(){return time;}public void run(){task.run();}
}//模拟实现一个定时器
class MyTimer{private PriorityQueue<MyTimerTask> queue = new PriorityQueue<>();private Object locker = new Object();public void schedule(Runnable task , long delay){synchronized (locker){MyTimerTask timerTask = new MyTimerTask(task,System.currentTimeMillis()+delay);queue.offer(timerTask);locker.notify();}}public MyTimer(){//创建一个线程,负责执行队列中的任务Thread worker = new Thread(() -> {try {while (true){synchronized (locker){//取出队首元素while(queue.isEmpty()){locker.wait();}MyTimerTask task = queue.peek();if(System.currentTimeMillis()<task.getTime()){locker.wait(task.getTime() - System.currentTimeMillis());}else {task.run();queue.poll();}}}}catch (InterruptedException e) {throw new RuntimeException(e);}},"worker");}
}
public class Demo33 {public static void main(String[] args) {MyTimer timer = new MyTimer();timer.schedule(new Runnable() {@Overridepublic void run() {System.out.println("hello 3000");}},3000);timer.schedule(new Runnable() {@Overridepublic void run() {System.out.println("hello 2000");}},2000);timer.schedule(new Runnable() {@Overridepublic void run() {System.out.println("hello 1000");}},1000);//Executors.newScheduledThreadPool(4);}}

     3.  拓展时间轮     



     4. 优先级队列 & 时间轮的优缺点    

     c96f743646e841f8bb30b2d242197f2f.gif

692a78aa0ec843629a817408c97a8b84.gif

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/886563.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

AIGC----生成对抗网络(GAN)如何推动AIGC的发展

AIGC: 生成对抗网络(GAN)如何推动AIGC的发展 前言 随着人工智能领域的迅猛发展&#xff0c;AI生成内容&#xff08;AIGC&#xff0c;AI Generated Content&#xff09;正成为创意产业和技术领域的重要组成部分。在AIGC的核心技术中&#xff0c;生成对抗网络&#xff08;GAN&am…

基于css的Grid布局和vue实现点击左移右移轮播过渡动画效果

直接上代码&#xff0c;以下代码基于vue2,需要Vue3或者react可以使用国内直连GPT/Claude来帮你转换下 代码如下&#xff1a; // ScrollCardsGrid.vue <template><div class"scroll-cards-container"><!-- 左箭头 --><div v-show"showLef…

Springboot如何打包部署服务器

文章目的&#xff1a;java项目打包成jar包或war包&#xff0c; 放在服务器上去运行 一、编写打包配置 1. pom.xml 在项目中的pom.xml文件里面修改<build>...</build>的代码 >> 简单打包成Jar形式&#xff0c;参考示例&#xff1a; <build><fina…

HTML、CSS与JavaScript基础

HTML&#xff1a;网页的骨架 互联网起源与HTML概述 互联网起源于1960年代的美国&#xff0c;HTML&#xff08;HyperText Markup Language&#xff09;作为构建网页的标准语言&#xff0c;自1990年代起便成为互联网信息展示的基石。 HTML文档结构 一个标准的HTML文档包含<…

实现了两种不同的图像处理和物体检测方法

这段代码实现了两种不同的图像处理和物体检测方法&#xff1a;一种是基于Canny边缘检测与轮廓分析的方法&#xff0c;另一种是使用TensorFlow加载预训练SSD&#xff08;Single Shot Multibox Detector&#xff09;模型进行物体检测。 1. Canny边缘检测与轮廓分析&#xff1a; …

Git 修改用户名(user.name)和用户邮件地址(user.email)的方法和作用

文章目录 修改方法修改作用 修改方法 首先&#xff0c;需要在本地计算机上打开Git Bash&#xff0c;然后确定你是只需要修改当前Git仓库的用户名和用户邮件地址&#xff0c;还是计算机上所有Git仓库的用户名和用户邮件地址。 对于只修改当前Git仓库的用户名和用户邮件地址的情…

RabbitMQ-死信队列(golang)

1、概念 死信&#xff08;Dead Letter&#xff09;&#xff0c;字面上可以理解为未被消费者成功消费的信息&#xff0c;正常来说&#xff0c;生产者将消息放入到队列中&#xff0c;消费者从队列获取消息&#xff0c;并进行处理&#xff0c;但是由于某种原因&#xff0c;队列中的…

ALSA - (高级Linux声音架构)是什么?

ALSA是Linux声音系统的核心组件&#xff0c;让用户可以精细控制声音硬件和声音进出。它通过抽象层屏蔽了硬件复杂性&#xff0c;使开发者能够专注于功能实现。这篇文章将逐步解析ALSA的基础知识&#xff0c;包括其运作原理、应用场景&#xff0c;以及如何完成一个基本配置和使用…

Ceph层次架构分析

Ceph的层次结构可以从逻辑上自下向上分为以下几个层次&#xff1a; 一、基础存储系统RADOS层 功能&#xff1a;RADOS&#xff08;Reliable Autonomic Distributed Object Store&#xff09;是Ceph的底层存储系统&#xff0c;提供了分布式存储的核心功能。它是一个完整的对象存…

常见error集合

Cannot use import statement outside a module 原因&#xff1a;在commonJS中用了es6的语法&#xff0c;import。分析&#xff1a; 一般我们的运行环境按照模块化标准来分&#xff0c;可以分为es6和commonJS两种&#xff0c;在es6中引入模块用import&#xff0c;在commonJS中…

在连锁零售行业中远程控制软件的应用

在连锁零售行业&#xff0c;远程控制软件正逐渐成为提高效率和降低成本的重要工具。作为零售经理&#xff0c;您可能已经注意到这种技术带来的变化。试想一下&#xff0c;无论您身在何处&#xff0c;都可以实时监控商店的运营情况&#xff0c;甚至在远离的地方解决顾客的问题。…

【MySQL】MySQL中的函数之JSON_REPLACE

在 MySQL 中&#xff0c;JSON_REPLACE() 函数用于在 JSON 文档中替换现有的值。如果指定的路径不存在&#xff0c;则 JSON_REPLACE() 不会修改 JSON 文档。如果需要添加新的键值对&#xff0c;可以使用 JSON_SET() 函数。 基本语法 JSON_REPLACE(json_doc, path, val[, path,…

JS学习日记(jQuery库)

前言 今天先更新jQuery库的介绍&#xff0c;它是一个用来帮助快速开发的工具 介绍 jQuery是一个快速&#xff0c;小型且功能丰富的JavaScript库&#xff0c;jQuery设计宗旨是“write less&#xff0c;do more”&#xff0c;即倡导写更少的代码&#xff0c;做更多的事&#xf…

Go语言24小时极速学习教程(五)Go语言中的SpringMVC框架——Gin

作为一个真正能用的企业级应用&#xff0c;怎么能缺少RESTful接口呢&#xff1f;所以我们需要尝试在Go语言环境中写出我们的对外接口&#xff0c;这样前端就可以借由Gin框架访问我们数据库中的数据了。 一、Gin框架的使用 1. 安装 Gin 首先&#xff0c;你需要在你的 Go 项目…

支持用户注册和登录、发布动态、点赞、评论、私信等功能的社交媒体平台创建!!!

需要整体源代码的可以在我的代码仓下载https://gitcode.com/speaking_me/social-media-platformTest.git 社交媒体平台 描述&#xff1a;社交媒体平台需要支持用户注册、发布动态、点赞、评论、私信等功能。 技术栈&#xff1a; 前端&#xff1a;React, Angular, Vue.js后端…

数字IC后端实现之Innovus specifyCellEdgeSpacing和ICC2 set_placement_spacing_rule的应用

昨天帮助社区IC训练营学员远程协助解决一个Calibre DRC案例。通过这个DRC Violation向大家分享下Innovus和ICC2中如何批量约束cell的spacing rule。 数字IC后端手把手实战教程 | Innovus verify_drc VIA1 DRC Violation解析及脚本自动化修复方案 下图所示为T12nm A55项目的Ca…

Spring Boot中的自动装配机制

文章目录 1. 什么是自动装配&#xff1f;2. 自动装配是如何工作的&#xff1f;3. 如何开启自动装配&#xff1f;4. 自动装配的注意事项5. 结语推荐阅读文章 在Spring Boot的世界里&#xff0c;自动装配&#xff08;Auto-configuration&#xff09;就像春风拂面&#xff0c;轻轻…

【时间之外】IT人求职和创业应知【37】-AIGC私有化

目录 新闻一&#xff1a;2024智媒体50人成都会议暨每经20周年财经媒体峰会召开 新闻二&#xff1a;全球机器学习技术大会在北京召开 新闻三&#xff1a;区块链技术在金融领域的应用取得新突破 不知不觉的坚持了1个月&#xff0c;按照心理学概念&#xff0c;还要坚持2个月&am…

基于语法树的SQL自动改写工具开发系列(2)-使用PYTHON进行简单SQL改写的开发实战

一、前言 前面一篇写了如何搭建环境&#xff0c;本文接着讲怎么使用antlr4进行开发。 二、实战 根据上一篇&#xff0c;基于语法树的SQL自动改写工具开发系列&#xff08;1&#xff09;-离线安装语法树解析工具antlr4-DA-技术分享-M版,先在本地部署好开发环境。 DEMO 1 写…

基于单片机智能温室大棚监测系统

本设计以单片机为核心的智能温室大棚监测系统&#xff0c;用于监测大棚内的温湿度、土壤湿度、CO2浓度和光照强度。该系统以STM32F103C8T6芯片为核心控制单元&#xff0c;涵盖电源、按键、NB-IoT模块、显示屏模块、空气温湿度检测、土壤湿度检测、二氧化碳检测和光敏电阻等模块…