《javaEE篇》--定时器

news2024/11/26 12:38:54

定时器概念

当我们不需要某个线程立刻执行,而是在指定时间点或指定时间段之后执行,假如我们要定期清理数据库里的一些信息时,如果每次都手动清理的话就太麻烦,所以就可以使用定时器。定时器就可以比作一个闹钟,可以让我们的线程在指定的时间执行,还可以指定时间循环执行。

标准库中的定时器

在标准库中提供了一个Timer类,Timer是一种定时器工具,可以让一个线程在指定时间一次或反复执行。

Timer的构造方法

Timer()
Timer(String  name)设定定时器的名字
Timer(boolean  isDeamon)是否将定时器作为守护线程
Timer(String  name,boolean  isDeamon)设定定时器名字,并设定是否为守护线程执行

非守护线程:JVM会等待所有非守护线程执行完毕之后才会退出

守护线程:JVM不会等待守护线程执行完毕,当没有非守护线程在执行JVM就会关闭,即使 有守护线程在执行

TimerTask是一个抽象类,表示的是一个可以被timer执行的任务,任务的具体实现在TimerTask的run方法里。

schedule方法是Timer中的核心方法用来执行TimerTask的任务并设定时间,具体的参数有以下几种

  • schedule(TimerTask task,Date time);

 在time时间(时间点)执行任务一次

  • schedule(TimerTask task,long delay);

在delay时间后执行任务一次(单位毫秒)

  • schedule(TimerTask task,Date firstTime,long period);

在firstTime时间点执行一次任务,在定期的period时间段后反复执行(如果时间点是已经过去的时间就会立刻执行)

  • schedule(TimerTask task,long delay,long period);

在延迟delay毫秒后执行任务一次,在定期的period时间段后反复执行(如果时间点是已经过去的时间就会立刻执行)

定时器的使用

在指定时间段后执行

public static void main(String[] args) {
        Timer timer = new Timer();//对象被创建时,线程也跟着被创建,后续timer执行任务都是由这个线程执行
        //给定时器安排一个任务,xxxms后执行
        timer.schedule(new TimerTask() {
            @Override
            public void run() {
                System.out.println("开始执行定时器任务");
            }
        },1000);
        System.out.println("执行任务");
    }

 主线程执行schedule方法的时候,就是把这个任务放到timer对象中了,与此同时timer里头也有一个线程叫做“扫描线程”,一旦时间到扫描线程就会执行刚才安排的任务了,仔细观察可以发现,整个线程,其实没有结束,就是因为Timer内部的线程阻止了进程的结束,而且在timer里是可以安排多个任务的,这些任务会按照时间顺序执行。(也就是我们刚刚说过的非守护线程,给timer设置成守护线程,此时整个线程就会结束了)

 在指定时间点执行

 public static void main(String[] args) {
        Timer timer = new Timer();//对象被创建时,线程也跟着被创建,后续timer执行任务都是由这个线程执行
        //给定时器安排一个任务,xxxms后执行

        //获取5秒后的一个时间点:doTime
        Calendar cal=Calendar.getInstance();
        cal.add(Calendar.SECOND,5);
        Date doTime=cal.getTime();
        timer.schedule(new TimerTask() {
            @Override
            public void run() {
                System.out.println("开始执行定时器任务" + new Date());
            }
        },doTime);
        System.out.println(new Date());
    }

在指定时间段后定期循环执行

public static void main(String[] args) {
        Timer timer = new Timer();//对象被创建时,线程也跟着被创建,后续timer执行任务都是由这个线程执行
        //给定时器安排一个任务,xxxms后执行

        timer.schedule(new TimerTask() {
            @Override
            public void run() {
                System.out.println("开始执行定时器任务" + new Date());
            }
        },2000,1000);
        System.out.println(new Date());
    }

在指定时间点后定期循环执行 

public static void main(String[] args) {
        Timer timer = new Timer();//对象被创建时,线程也跟着被创建,后续timer执行任务都是由这个线程执行
        //给定时器安排一个任务,xxxms后执行

        //获取5秒后的一个时间点:doTime
        Calendar cal=Calendar.getInstance();
        cal.add(Calendar.SECOND,5);
        Date doTime=cal.getTime();
        timer.schedule(new TimerTask() {
            @Override
            public void run() {
                System.out.println("开始执行定时器任务" + new Date());
            }
        },doTime,1000);
        System.out.println(new Date());
    }

定时器的实现

这里我们来简单实现一个定时器

初步分析

  1. Timer中要有一个扫描线程线程,扫描任务是否到时间可以执行了
  2. 需要一种数据结构来储存这些任务
  3. 还需要创建一个类来描述一个任务,要包含内容和时间

 先来考虑使用什么数据结构来储存任务,既然Timer中的任务是从时间最小的开始执行,那么我们就可以使用优先级队列!!

实现

MyTimerTask

一个执行的任务至少要包含任务内容和执行的时间,这里我用的是绝对时间当要判断一个线程是否需要执行时,先获取一个当前的时间戳(2:30),在获取任务要执行的时间戳(2:35),最后对比两个时间戳就知道现在是否要执行任务。

但是我们刚刚有提到,要使用优先级队列来存储任务,所以我们还要有一个任务的比较规则(我们要让队列知道该怎么比)

所以完整的代码如下:

class MytimerTask implements Comparable<MytimerTask>{
    //要执行的任务
    private Runnable runnable;
    //要执行任务的绝对时间
    private long time;
    public MytimerTask(Runnable runnable,long time){
        this.runnable = runnable;
        //此处this.time是应该开始执行任务的时间,time是多久后执行
        this.time = System.currentTimeMillis() + time;
    }
    //任务的比较规则
    @Override
    public int compareTo(MytimerTask o) {
        return (int)(this.time - o.time);
    }

    public Runnable getRunnable() {
        return runnable;
    }

    public long getTime() {
        return this.time;
    }
}

MyTimer

定义一个优先级队列,实现schedule方法

 private PriorityQueue<MytimerTask> queue = new PriorityQueue<>();

    public void schedule(Runnable runnable,long time){  
            //向队列里插入一个任务,以及过多久后执行该任务
            queue.offer(new MytimerTask(runnable,time));
    }

实现扫描线程 

private Object locker = new Object();
    public MyTimer(){
        //创建一个扫描线程
        Thread t = new Thread(() -> {
            //扫描线程,需要不停扫描队首元素,看是否到达时间
            while (true){
                 try {
                     synchronized (locker){
                         //使用while是为了在wait被唤醒的时候,再次确认一下条件
                         while (queue.isEmpty()){
                             //这里的wait,需要有另外线程唤醒
                             //添加新的任务就应该唤醒
                             locker.wait();
                         }
                         MytimerTask task = queue.peek();
                         //比较一下当前的队首元素(就是最近一个要执行的任务)是否可以执行了
                         long curtime = System.currentTimeMillis();
                         if(curtime >= task.getTime()){
                             //当时间到达任务时间,就可以执行任务了
                             task.getRunnable().run();
                             //任务执行完之后在队列中删除
                             queue.poll();
                         }else {
                             //还没到任务时间,暂时不执行任务
                             locker.wait(task.getTime() - curtime);
                         }
                     }
                 }catch (InterruptedException e){
                     e.printStackTrace();
                     break;
                 }
            }
        });
        t.start();
    }

因为我们需要不断的判断队列里的线程是否到达时间,所以使用while循环

因为当前可能会是在多线程下使用,所以我们要保证线程安全问题,使用synchronization来进行加锁

如果队列为空的话,就需要阻塞等待直到有新的任务加入,所以schedule方法应该这样实现

public void schedule(Runnable runnable,long time){
        synchronized (locker){
            //向队列里插入一个任务,以及过多久后执行该任务
            queue.offer(new MytimerTask(runnable,time));
            //当队列为空时,此时插入一个任务之后唤醒wait
            locker.notify();
        }
    }

这里还有一个问题, 因为while的执行速度很快,如果让他一直不断的循环,就会造成忙等问题(在消耗cpu资源但是并没有实质性的用处),假如最近一个要执行的任务是十分钟之后,那么从现在开始的十分钟之内,while是没有执行的必要的,不停的循环只会浪费资源。

所以我们可以进行判断,当还没有到最近一个任务执行时间时,我们就让线程阻塞,阻塞的时间就是现在到最近一个要执行的任务的时间,这样就可以解决忙等问题。

那么可不可以使用sleep呢?当然是不行的。

可能在等待过程中主线程调用schedule又添加一个新任务,这个任务比其他任务执行的时间更早,使用wait就恰好可以利用刚才schedule中的notify来唤醒这里的wait,让循环再执行一遍,重新拿到队首执行时间最少的任务

之所以刚刚的代码使用的是PriorityQueue而不是PriorityBlockingQueue,其实就是因为要处理两个wait的地方,使用阻塞版本的优先级队列,不方便实现。 

 完整代码如下

class MytimerTask implements Comparable<MytimerTask>{
    //要执行的任务
    private Runnable runnable;
    //要执行任务的绝对时间
    private long time;
    public MytimerTask(Runnable runnable,long time){
        this.runnable = runnable;
        //此处this.time是应该开始执行任务的时间,time是多久后执行
        this.time = System.currentTimeMillis() + time;
    }
    //任务的比较规则
    @Override
    public int compareTo(MytimerTask o) {
        return (int)(this.time - o.time);
    }

    public Runnable getRunnable() {
        return runnable;
    }

    public long getTime() {
        return this.time;
    }
}
//自己的定时器
class MyTimer{
    private PriorityQueue<MytimerTask> queue = new PriorityQueue<>();

    public void schedule(Runnable runnable,long time){
        synchronized (locker){
            //向队列里插入一个任务,以及过多久后执行该任务
            queue.offer(new MytimerTask(runnable,time));
            //当队列为空时,此时插入一个任务之后唤醒wait
            locker.notify();
        }
    }
    private Object locker = new Object();
    public MyTimer(){
        //创建一个扫描线程
        Thread t = new Thread(() -> {
            //扫描线程,需要不停扫描队首元素,看是否到达时间
            while (true){
                //开始进行一次扫描
                 try {
                     synchronized (locker){
                         //使用while是为了在wait被唤醒的时候,再次确认一下条件
                         while (queue.isEmpty()){
                             //这里的wait,需要有另外线程唤醒
                             //添加新的任务就应该唤醒
                             locker.wait();
                         }
                         MytimerTask task = queue.peek();
                         //比较一下当前的队首元素(就是最近一个要执行的任务)是否可以执行了
                         long curtime = System.currentTimeMillis();
                         if(curtime >= task.getTime()){
                             //当时间到达任务时间,就可以执行任务了
                             task.getRunnable().run();
                             //任务执行完之后在队列中删除
                             queue.poll();
                         }else {
                             //还没到任务时间,暂时不执行任务
                             locker.wait(task.getTime() - curtime);
                         }
                     }
                 }catch (InterruptedException e){
                     e.printStackTrace();
                     break;
                 }
            }
        });
        t.start();
    }
}

到这里一个简单的定时器就完成了 

测试

  public static void main(String[] args) {
        MyTimer timer = new MyTimer();
        timer.schedule(new Runnable() {
            @Override
            public void run() {
                System.out.println("3000" + new Date());
            }
        },3000);
        timer.schedule(new Runnable() {
            @Override
            public void run() {
                System.out.println("2000"+ new Date());
            }
        },2000);
        timer.schedule(new Runnable() {
            @Override
            public void run() {
                System.out.println("1000"+ new Date());
            }
        },1000);
        System.out.println("启动程序");
    }

以上就是博主对定时器知识的分享,如果有不懂的或者有其他见解的欢迎在下方评论或者私信博主,也希望多多支持博主之后和博客!!🥰🥰

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

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

相关文章

C++ 设计模式(6. 适配器模式)

适配器模式Adapter Pattern是一种结构型设计模式&#xff0c;它可以将一个类的接口转换成客户希望的另一个接口&#xff0c;主要目的是充当两个不同接口之间的桥梁&#xff0c;使得原本接口不兼容的类能够一起工作。基本结构 Target 是目标接口&#xff0c;Adaptee 是需要适配的…

微信小程序实例代码解读

以微信 小程序开发工具给的示例代码为例&#xff1a; 主页代码&#xff1a; index.wxml 这个文件是一个微信小程序页面的 WXML 结构,主要功能是展示一个快速开始教程的步骤和内容。 源代码&#xff1a; <!--index.wxml--> <view class"container">&l…

ZK-Rollups测评

1. 引言 Matter Labs团队和多个高校研究人员一起&#xff0c;发布2024年论文《Analyzing and Benchmarking ZK-Rollups》&#xff0c;开源代码见&#xff1a; https://github.com/StefanosChaliasos/zkrollup-benchmarking&#xff08;Python&#xff09; 其中&#xff1a; …

安装MySQL入门基础指令

一.安装MySQL(以5.7版本为例) 1.一路默认安装&#xff0c;截图供大家参考 修改自己window安装名字即可 2.配置环境变量 C:\Program Files\MySQL\MySQL Server 5.7\bin 写入系统环境变量即可在window窗口使用其服务了 3.登录MySQL服务 进入控制台输入命令 mysql -u root …

运维小技能:基于Windows系统和‌Linux系统,以tomcat为案例,讲解如何新增自启动服务。

文章目录 引言‌I Linux系统‌(以CentOS为例)基础知识:运行级别(run level)基于chkconfig 工具,设置服务启动类型。基于systemctl 新增系统服务II 基于Windows系统设置服务自启动的常规操作安装多个tomcat服务,并设置自启动。III 扩展制定定时任务优化停止Tomcat服务命令引…

ESP32Cam人工智能教学20

ESP32Cam人工智能教学20 ESP32Cam专用APP 这次我们专门为ESP32Cam量身定制一个手机APP。手机APP是客户端&#xff0c;利用Socket连接ESP32Cam&#xff0c;ESP32Cam成了服务器&#xff0c;实现Socket全双工的数据传输模式&#xff0c;还可以一边显示摄像头图像&#xff0c;一边…

【Canvas与诗词】北岛诗《献给遇罗克》节选(以太阳的名义...)

【成图】 【代码】 <!DOCTYPE html> <html lang"utf-8"> <meta http-equiv"Content-Type" content"text/html; charsetutf-8"/> <head><title>以太阳的名义</title><style type"text/css">…

基类没有虚析构,即使派生类使用智能指针也一定会内存泄漏

实验 定义一个基类和一个派生类 class Base { public:virtual ~Base() default; };class Derive :public Base { public:std::shared_ptr<int> sp{new int{0},[](int *p){delete p;std::cout << "删除器" << endl;},}; };main 函数执行如下代码…

作业08.21

服务器&#xff1a; #include <myhead.h>#define SER_PORT 6666 #define SER_IP "127.0.0.1"int find_client(int *client_arr, int len, int client) {for(int i0; i<len; i){if(client_arr[i] client){return i;}}return -1; }void remove_client(int *…

Mac 使用vscode 创建vue项目后修改文件提示:权限不足,以超级用户身份重试

项目场景&#xff1a; Mac 安装了全局 vue-cli 插件后&#xff0c;使用webpack 创建vue项目&#xff0c;打开项目&#xff0c;选择信任所有文件夹&#xff0c;然后正常编写代码&#xff0c;并对项目中的文件进行修改&#xff0c;点击保存的时候提示&#xff1a;保存“webpack.…

Vue3+Ts封装类似el-dialog的对话框组件

提供11个字段对dialog组件进行控制&#xff1a; modelValue: 对话框显示隐藏控制, width: 控制对话框的宽度, height&#xff1a;控制对话框的高度, top: 控制对话框个距离顶部的距离, title: 控制对话框的标题, appendToBody: 是否将对话框添加至body, closeOnClickModa…

GX Works2的使用方法

目录&#xff1a; 1、概述 2、硬件连接 3、录入与修改程序 1&#xff09;进入编辑按F2或点击“写入模式”图标 2&#xff09;修改部分元件 3&#xff09;注释 4&#xff09;改变显示触点数 4、软仿真与在线仿真 1&#xff09;软仿真 2&#xff09;在线仿真 5、P…

Linux源码阅读笔记-USB设备驱动架构

总线速度及主机控制器 USB系统架构 USB系统主机端提供为4个引脚的A型接口&#xff0c;USB外围设备通过4个引脚的B型接口和主机端连接。那4个引脚&#xff08;一条电压线VBUS、一条地线GND、一条正方向传输数据的D和一条反方向传输数据的D-线。&#xff09;USB主机和USB设备收发…

2024年翻译神器:探索四款好用的翻译工具!

因为有了一些翻译工具的存在&#xff0c;语言障碍已经渐渐不成问题。接下来就为大家推荐几款好用的翻译工具&#xff01; 福昕在线翻译 链接&#xff1a; https://fanyi.pdf365.cn/ 福昕在线翻译以其简洁的界面和强大的翻译能力&#xff0c;成为用户跨越语言障碍的首选。它…

独立站PrestaShop安装

独立站PrestaShop安装 独立站PrestaShop安装系统需求下载PrestaShop浏览器下载命令行下载 解压PrestaShop创建数据库移动PrestaShop源码到web目录composer安装依赖包nginx配置访问域名进入安装页面选择语言许可协议系统兼容性店铺信息Content of your store系统配置数据库店铺安…

金矢之lian,非你莫蜀:金矢留学携手16所英国大学共襄成都盛会

碧海蓝天的东海岸&#xff0c;盛夏的热烈未尽&#xff0c;草书云山如锦绣的天府之国&#xff0c;初秋的凉意渐起。近一年的忙碌与等待之后&#xff0c;2024年英国秋季入学申请已近尾声&#xff0c;如愿以偿拿到了录取的同学们&#xff0c;欢欣鼓舞的进入申请签证甚至预定行程机…

nginx实例

nginx的由来 Nginx是由1994年毕业于俄罗斯国立莫斯科鲍曼科技大学的同学为俄罗斯rambler.ru公司开发的&#xff0c;开发工作最早从2002年开始&#xff0c;第一次公开发布时间是2004年10月4日&#xff0c;版本号是0.1.0。2019年3月11日F5 与 NGINX达成协议,F5 将收购 NGINX 的所…

NSSCTF联系记录:[SWPUCTF 2021 新生赛]crypto7

题目&#xff1a; 一共有32个字符&#xff0c;且只有数字和字母&#xff0c;可能为md5加密 得到答案

Linux shell编程学习笔记74:sed命令——沧海横流任我行(中)

0 前言 自 60 年代末以来&#xff0c;sed 一直是 Unix 标准工具箱的一部分。 Sed在以下三种情况下特别有用&#xff1a; 编辑太大的文件&#xff0c;无法进行舒适的交互式编辑&#xff1b; 当编辑命令序列过于复杂而无法在交互模式下轻松键入时&#xff0c;可以编辑任何大小的…