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

news2025/1/4 18:44:14

目录

一. 什么是“定时器”

二. 标准库的定时器 

三. 定时器的实现 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() {
            @Override
            public void run() {
                System.out.println("hello delay"+" " +1000);
            }
        },1000);
        timer1.schedule(new TimerTask() {
            @Override
            public void run() {
                System.out.println("hello delay"+" " +2000);
            }
        },2000);
        timer1.schedule(new TimerTask() {
            @Override
            public 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;
    }

    @Override
    public 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);
            // 唤醒 wait
            locker.notify();
        }
    }
}

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

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

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

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


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

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


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

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

四. 拓展

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

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

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

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

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

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

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

相关文章

【Spring】详细理解Spring中控制反转(IOC)和依赖注入(DI)的设计思想。

目录 1.相关概念 2. 设计IOC容器的两个主要接口 2.1 BeanFactory接口 2.2 ApplicationContext接口 2.3 两个接口的区别【面试题】 3. Bean对象的管理&#xff08;控制反转&#xff09; 3.1 基于XML文件管理bean对象 3.2 基于注解方式去管理【重点】 4. 依赖注入&#…

81.WEB渗透测试-信息收集-框架组件识别利用(5)

免责声明&#xff1a;内容仅供学习参考&#xff0c;请合法利用知识&#xff0c;禁止进行违法犯罪活动&#xff01; 内容参考于&#xff1a; 易锦网校会员专享课 上一个内容&#xff1a;80.WEB渗透测试-信息收集-框架组件识别利用&#xff08;4&#xff09; solr&#xff1a;…

第三方库jsoncpp

文章目录 0.jsoncpp库是做什么的&#xff1f;1.安装库2.有哪几个类&#xff0c;如何使用Json::Value类Json::Writer类-StreamWriterJson::Reader类-CharReader三者关系 3.使用样例将数据先存入Value类&#xff0c;再通过StreamWriter类转为Json格式的字符串获取到Json格式字符串…

6 postgresql事务与并发控制

事务与并发控制 事务可以理解为做一件事&#xff0c;数据库就是一件事要么做完&#xff0c;要么不做&#xff0c;不然这数据库给给人不可靠的感觉 正如解释事务具有四个重要特性&#xff1a;ACID&#xff08;原子性&#xff0c;一致性&#xff0c;隔离性&#xff0c;持久性&a…

python实现提取视频帧的图片

文章目录 1、需求痛点2、完整代码⭐3、代码分析3.1、需要改动的地方3.2、OpenCV库的使用3.3、多线程技术 4、执行效率5、效果展示⭐6、注意事项&#x1f53a;7、总结 &#x1f343;作者介绍&#xff1a;双非本科大三网络工程专业在读&#xff0c;阿里云专家博主&#xff0c;专注…

3 IIC总线

3 IIC总线 1、基本概念1.1 IIC总线定义1.2 IIC总线协议概念 2 以AT24C02为例说明时序2.1 基本特性2.2 利用GPIO模拟IIC2.3 对AT24C02的操作2.4 重定向printf 1、基本概念 1.1 IIC总线定义 定义&#xff1a;两线式串行总线 两线式&#xff1a;说明处理器和外设之间只需两根信号…

【时间复杂度和空间复杂度】(内含超多实例练习)

【时间复杂度和空间复杂度】&#xff08;内含超多实例练习&#xff09; 1. 算法效率2. 时间复杂度2.1 时间复杂度的概念2.2 实例练习2.2.1 数组中搜索数据2.2.2 冒泡排序2.2.3 二分查找2.2.4 阶乘递归2.2.5 斐波那契递归 3. 空间复杂度3.1 空间复杂度的概念3.2 实例练习3.2.1 冒…

Set Hashset底层原理 LinkedSet底层原理 Treeset 31

不重复只执行一次&#xff0c;排序是小到大&#xff0c;色图是接口类 Hashset底层原理 LinkedSet底层原理 Treeset 总结

AI+云边端协同,EasyCVR视频汇聚技术赋能安防监控新生态

随着信息技术的飞速发展和数字化时代的到来&#xff0c;安防监控领域的技术也在不断创新和突破。EasyCVR平台的视频汇聚技术作为其中的佼佼者&#xff0c;以其强大的视频处理、汇聚与融合能力&#xff0c;在安防监控领域展现出了巨大的应用潜力和价值。本文将详细介绍EasyCVR视…

权限束缚术:windows工具自动化权限提升

前言 欢迎来到我的博客 个人主页:北岭敲键盘的荒漠猫-CSDN博客 本文整理利用工具全自动化提权的相关操作方法 因为我没有相关的环境复现&#xff0c;所以没有成功 不过复现的操作就是下面整理的操作 溢出漏洞提权原理 利用溢出漏洞提权&#xff0c;实际上就是一次判断过往的…

极简聊天室-websocket版

再写一个极简聊天室的websocket版&#xff0c;在本例中&#xff0c;websocket仅用于服务器向客户端传输信息&#xff0c;客户端向服务器发送信息是传统的http post方式&#xff0c;用axios来实现的&#xff0c;当然websocket本身是支持双向通信&#xff0c;主要是为了方便跟前面…

Selenium 无法定位元素的几种解决方案

&#x1f345; 点击文末小卡片&#xff0c;免费获取软件测试全套资料&#xff0c;资料在手&#xff0c;涨薪更快 1、frame/iframe表单嵌套 WebDriver只能在一个页面上对元素识别与定位&#xff0c;对于frame/iframe表单内嵌的页面元素无法直接定位。 解决方法&#xff1a; d…

LLM大模型实战项目--基于ChatGLM2的小书虫文档阅读助手

本文介绍如此从零开始编写一个基于ChatGLM2的文档阅读助手 一、项目介绍 二、获取API接口 三、环境配置 四、代码实现 五、效果展示 一、项目介绍 小书虫&#x1f4da; 文档阅读助手是一个基于百度千帆大模型平台的Web应用程序&#xff0c;核心是清华大学训练的ChatGLM2大模…

算法加项目

1.仅加号 很简单&#xff0c;找到每次添加后最小的即可&#xff0c;这里不做演示。 2.采药 dp&#xff0c;for1数目&#xff0c;for2时间&#xff0c;简单的dp模板题目。 #include<bits/stdc.h> using namespace std; int n,m; const int N105,M105; long long a[M],b[M…

TInyWebServer面试题

一、项目介绍 &#xff08;1&#xff09;为什么要做这样一个项目&#xff1f; &#xff08;2&#xff09;介绍一下你的项目 这个项⽬是我在学习计算机⽹络和Linux socket编程过程中独⽴开发的轻量级Web服务器&#xff0c;服务器的⽹络模型是主从reactor加线程池的模式&#xf…

VMware虚拟机网络模式配置详解【原理,功能,特点层面】

VMware虚拟机网络模式配置详解【原理,功能,特点层面】 文章目录 VMware虚拟机网络模式配置详解【原理,功能,特点层面】桥接模式&#xff08;Bridged&#xff09;原理功能点介绍虚拟网络编辑器配置虚拟机配置 配置教程编辑-虚拟网络编辑器虚拟机网络配置 特点 NAT模式概念功能点…

数学建模--智能算法之蚁群优化算法

目录 基本原理 算法步骤 Python代码示例 应用领域 特点及改进 蚁群优化算法在解决哪些具体组合优化问题方面表现最为突出&#xff1f; 如何有效地改进蚁群优化算法以提高其收敛速度和避免陷入局部最优的问题&#xff1f; 蚁群优化算法与其他群体智能优化算法&#xff0…

三维旋转矩阵

前言 本文讲述三维旋转的矩阵推导&#xff0c;推导过程遵循下面的规则&#xff1a; 本文的坐标系是基于右手坐标系的逆时针旋转为正向旋转 围绕坐标轴的旋转 x x x轴 我们假设旋转的点为 P P P 假设旋转之前点 P P P的坐标为 ( x 0 , y 0 , z 0 ) (x_0,y_0,z_0) (x0​,y0​,…

CAN总线中注入拓展帧、远程帧,CAPL通用函数。

🍅 我是蚂蚁小兵,专注于车载诊断领域,尤其擅长于对CANoe工具的使用🍅 寻找组织 ,答疑解惑,摸鱼聊天,博客源码,点击加入👉【相亲相爱一家人】🍅 玩转CANoe,博客目录大全,点击跳转👉 【测试需求】:CAN总线要求DUT 接收到非预期的扩展帧、远程帧时,通信正常,…

vulhub:nginx解析漏洞nginx_parsing

这个解析漏洞其实是PHP CGI的漏洞&#xff0c;在PHP的配置文件中有一个关键的选项cgi.fix_pathinfo默认是开启的&#xff0c;当URL中有不存在的文件&#xff0c;PHP就会向前递归解析。在一个文件/xx.jpg后面加上/.php会将 /xx.jpg/xx.php 解析为 php 文件 条件&#xff1a; Ngi…