网站首页 > 技术文章 正文
标准库中的Timer
标准库中有一个Timer类,java.util.Timer,核心方法为schedule,schedule有两个参数,第一个参数为即将要执行的任务,第二个参数为多久后执行该任务(单位为毫秒),任务为new TimerTask(),TimerTask为抽象类,实现了Ruannable接口,具体看一下使用
import java.util.Timer;
import java.util.TimerTask;
public class Demo {
public static void main(String[] args) {
//Timer内部是专门有线程来执行我们注册的任务,这个线程在执行完一个任务还会等待别的任务执行
Timer timer = new Timer(); //schedule(任务,多久后执行任务)
//TimerTask是一个抽象类,实现了Runnable接口
timer.schedule(new TimerTask() {
@Override
public void run() {
System.out.println("hello timer");
}
}, 3000);
System.out.println("main");
}
}
运行结果:先打印出main,3秒之后打印hello Timer
上述代码执行完,发现程序没有结束,原因是Timer内部是专门有线程来执行我们注册的任务,这个线程在执行完一个任务还会等待别的任务执行
模拟实现Timer
通过上述标准库中的Timer分析Timer内部需要啥东西
- 描述任务:创建一个类专门表示定时器中的一个任务
- 组织任务:使用数据结构来组织
- 执行时间到了的任务:创建定时器实例时,创建一个线程专门来执行此任务
描述任务
下面组织任务用到了优先级队列,优先级队列必须插入可以比较大小的元素,所以这里的任务类就必须实现比较器接口Comparable并重写compareTo方法,使得可以通过时间来进行比较大小,定时器在使用的时候需要获取时间最小的任务的时间,以此时间戳和当前时间戳比较看是否可以执行任务,所以此处也要提供getTime方法
//描述任务
class MyTask implements Comparable<MyTask>{
//任务具体的内容
private Runnable runnable;
//任务执行的时间戳 private long time;
//delay为时间间隔,不是具体的时间戳
public MyTask(Runnable runnable, long delay){
this.runnable = runnable;
this.time = System.currentTimeMillis()+delay;
}
@Override
public int compareTo(MyTask o) {
return (int) (this.time-o.time);
}
public void run(){
runnable.run();
}
public long getTime() {
return time;
}
}
组织任务
现在有多个任务,比如一个小时后做作业,半个小时后吃饭…,定时器在执行任务的时候,按照时间顺序先后顺序执行的,所以我们需要在安排的所有任务中找出距离要执行任务时间最短的任务,依次类推,不难得出,可以使用优先级队列这一数据结构来组织任务
注意: 此处的优先级队列要考虑线程安全问题,因为可能多个线程进行注册任务,还有一个专门的线程来执行任务,所以使用PriorityBlockingQueue
这里创建了一个对象用于加锁,具体原因在下面介绍
private PriorityBlockingQueue<MyTask> p = new PriorityBlockingQueue<>();
//创建一个对象用于加锁
private Object locker = new Object();
public void schedule(Runnable runnable, long delay){
MyTask task = new MyTask(runnable, delay);
p.put(task);
//插入任务,可能执行时间已经过了,需要唤醒等待的线程进行判断是否执行
synchronized (locker){
locker.notify();
}
}
执行时间到了的任务
需要有一个线程不停的检查优先级队列队头元素,判断该元素的执行时间是不是到了,所以在定时器的构造方法中创建一个线程来执行任务
public MyTimer(){
Thread t = new Thread(new Runnable() {
@Override
public void run() {
while(true){
try {
MyTask task = p.take();
if(task.getTime() > System.currentTimeMillis()){
p.put(task);
//当执行时间没到时,没必要一直进行判断,比较耗费CPU
//所以等待一定时间
synchronized (locker){
locker.wait(task.getTime()-System.currentTimeMillis());
}
}else {
task.run();
}
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
});
t.start();
}
为何等待使用wait和notify,而不使用sleep?
在任务的执行时间未到之前,可能判断次数很多,比较耗费CPU,而且没有必要一值判断,只需在一定时间内进行判断执行时间到没到即可,所以在还没有到执行时间时,使用wait(时间)来让该线程进行等待,在创建任务时唤醒等待即可,因为新的任务可能需要在刚才等待执行任务之前执行,也就是新创建的任务执行时间已经到了,所以要使用notify唤醒执行任务的线程继续进行判断时间是否执行,而且这个原因也是使用wait不使用sleep的原因,如果使用sleep,在新创建任务的执行时间在sleep等待结束时间之前,等待的线程没有办法唤醒,也就不能执行时间到了的任务。
作者:终有救赎
链接:https://juejin.cn/post/7290813210277494819
猜你喜欢
- 2024-09-20 非常详细!如何理解表格存储的多版本、生命周期和有效版本偏差
- 2024-09-20 6种快速统计代码执行时间的方法,真香
- 2024-09-20 Java 开发者最困惑的四件事(java开发遇到问题如何解决)
- 2024-09-20 还在用new Date计算任务执行时间?强烈建议使用这个API
- 2024-09-20 “抄”代码,再也不用上谷歌复制粘贴了
- 2024-09-20 java获取当前时间的四种方法代码实例
- 2024-09-20 撸完这篇线程池,我快咳血了(线程池有什么用)
- 2024-09-20 JAVA轮询遍历两个数组进行比较(遍历数组 java)
- 2024-09-20 蒙圈了?System.currentTimeMillis()存在性能问题
- 2024-09-20 .NET 9 中的 Task.WhenEach(.net task thread)
- 1514℃桌面软件开发新体验!用 Blazor Hybrid 打造简洁高效的视频处理工具
- 563℃Dify工具使用全场景:dify-sandbox沙盒的原理(源码篇·第2期)
- 508℃MySQL service启动脚本浅析(r12笔记第59天)
- 486℃服务器异常重启,导致mysql启动失败,问题解决过程记录
- 485℃启用MySQL查询缓存(mysql8.0查询缓存)
- 465℃「赵强老师」MySQL的闪回(赵强iso是哪个大学毕业的)
- 445℃mysql服务怎么启动和关闭?(mysql服务怎么启动和关闭)
- 442℃MySQL server PID file could not be found!失败
- 最近发表
- 标签列表
-
- c++中::是什么意思 (83)
- 标签用于 (65)
- 主键只能有一个吗 (66)
- c#console.writeline不显示 (75)
- pythoncase语句 (81)
- es6includes (73)
- windowsscripthost (67)
- apt-getinstall-y (86)
- node_modules怎么生成 (76)
- c++int转char (75)
- static函数和普通函数 (76)
- el-date-picker开始日期早于结束日期 (70)
- js判断是否是json字符串 (67)
- checkout-b (67)
- c语言min函数头文件 (68)
- asynccallback (71)
- localstorage.removeitem (74)
- vector线程安全吗 (70)
- & (66)
- java (73)
- js数组插入 (83)
- mac安装java (72)
- eacces (67)
- 查看mysql是否启动 (70)
- 无效的列索引 (74)