《javaEE篇》--定时器

embedded/2024/10/18 12:27:26/

定时器概念

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

标准库中的定时器

在标准库中提供了一个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时间段后反复执行(如果时间点是已经过去的时间就会立刻执行)

定时器的使用

在指定时间段后执行

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

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

 在指定时间点执行

java"> public static void main(String[] args) {Timer timer = new Timer();//对象被创建时,线程也跟着被创建,后续timer执行任务都是由这个线程执行//给定时器安排一个任务,xxxms后执行//获取5秒后的一个时间点:doTimeCalendar cal=Calendar.getInstance();cal.add(Calendar.SECOND,5);Date doTime=cal.getTime();timer.schedule(new TimerTask() {@Overridepublic void run() {System.out.println("开始执行定时器任务" + new Date());}},doTime);System.out.println(new Date());}

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

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

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

java">public static void main(String[] args) {Timer timer = new Timer();//对象被创建时,线程也跟着被创建,后续timer执行任务都是由这个线程执行//给定时器安排一个任务,xxxms后执行//获取5秒后的一个时间点:doTimeCalendar cal=Calendar.getInstance();cal.add(Calendar.SECOND,5);Date doTime=cal.getTime();timer.schedule(new TimerTask() {@Overridepublic void run() {System.out.println("开始执行定时器任务" + new Date());}},doTime,1000);System.out.println(new Date());}

定时器的实现

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

初步分析

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

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

实现

MyTimerTask

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

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

所以完整的代码如下:

java">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;}//任务的比较规则@Overridepublic int compareTo(MytimerTask o) {return (int)(this.time - o.time);}public Runnable getRunnable() {return runnable;}public long getTime() {return this.time;}
}

MyTimer

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

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

实现扫描线程 

java">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方法应该这样实现

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

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

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

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

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

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

 完整代码如下

java">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;}//任务的比较规则@Overridepublic 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));//当队列为空时,此时插入一个任务之后唤醒waitlocker.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();}
}

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

测试

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

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


http://www.ppmy.cn/embedded/100333.html

相关文章

Image Stride(内存图像行跨度)

When a video image is stored in memory, the memory buffer might contain extra padding bytes after each row of pixels. The padding bytes affect how the image is store in memory, but do not affect how the image is displayed. 当视频图像存储在内存时&#xff0…

object.defineProperty用法

Object.defineProperty 是 JavaScript 中一个用于定义对象属性的静态方法。它允许你精确控制对象的属性&#xff0c;包括属性的值、可写性、可枚举性和可配置性等特性。这个方法在需要定义特殊的属性行为时非常有用&#xff0c;例如&#xff0c;在 Vue.js 中&#xff0c;它常用…

OD C卷 - Wonderland游乐园

Wonderland游乐园&#xff08;200&#xff09; 游乐园有四种售票方式&#xff0c;分别为一日票&#xff0c;三日票&#xff0c;周票&#xff08;7天&#xff09;、月票&#xff08;30天&#xff09;&#xff1b;每种票据在时限内可以无限制游玩&#xff0c;如第10天买了一个三…

通过建模走出人工智能寒冬

很多人对 GenAI 是否会产生商业影响持怀疑态度&#xff0c;但我认为他们不仅错了&#xff0c;而且犯了 2001 年人们在互联网上犯下的错误。他们认为硅谷的炒作是无稽之谈&#xff0c;因此其背后的想法也是无稽之谈。 这是很危险的&#xff0c;我认为&#xff0c;这比大多数零售…

DC-DC开关电源稳压电路设计——7- 40V转换5V和3.3V

本篇文章记录分享DC-DC开关电源稳压&#xff08;7-40V转换5V和3.3V&#xff09;电路设计的思路及原理图。 目录 一、电路稳压原理图 二、开关稳压芯片 1、BUCK降压电路 2.LM2596 &#xff08;1&#xff09;、LM2596简介 &#xff08;2&#xff09;、LM2596原理 1. 基…

BAT32G137国产项目通用第五节:FreeRTOS 互斥量

主题:互斥量的使用比较单一,因为它是信号量的一种,并且它是以锁的形式存在。在初始 化的时候,互斥量处于开锁的状态,而被任务持有的时候则立刻转为闭锁的状态。互斥量 更适合于: 1.可能会引起优先级翻转的情况。 递归互斥量更适用于。 2.任务可能会多次获取互斥量的情…

vs code中编写html的配置,插件安装

首先安装vs code 插件安装下面三个&#xff1a; 功能分别是&#xff1a; html css support &#xff1a;就是支持html环境&#xff0c;因为vs code就是一个文本编辑器 live server&#xff1a;自动更新编写的文件在浏览器刷新 auto rename tag&#xff1a;自动修改另一半标签…

Python技能进阶:探索Selenium库,实现网页自动化测试与爬虫

一、引言 在数字化时代&#xff0c;网页自动化测试与爬虫成为了许多开发者必备的技能之一。Python作为一门功能强大的编程语言&#xff0c;拥有许多优秀的库可以帮助我们实现这一目标。其中&#xff0c;Selenium库以其强大的功能和广泛的应用领域&#xff0c;受到了广大开发者…