JavaEE 定时器
sugar high 人气:0前言
像是一个闹钟定时,在一定时间之后被唤醒并执行某个之前设定好的任务,join(指定超时时间),sleep(指定休眠时间)都是基于系统内部的定时器来实现的。
java.util.Timer核心方法就一个,schedule参数有两个:任务是啥(一段代码),多长时间之后执行
public class 定时器 { public static void main(String[] args) { Timer timer = new Timer(); timer.schedule(new TimerTask() { @Override public void run() { System.out.println("hello timer"); } }, 3000); } }
Timer内部组成
1.描述任务
创建一个专门的类来表示一个定时器的任务TimerTask,这个MyTask类的比较规则不是默认存在的,需要我们手动指定按照时间大小来比较的。
//创建一个类表示一个任务 class MyTask{ //任务描述 private Runnable runnable; //任务具体啥时候干 private long time; //after是一个时间间隔,不是觉得时间戳的值 public MyTask(Runnable runnable, long after){ this.runnable = runnable; this.time = System.currentTimeMillis() + after; } public void run(){ runnable.run(); } }
2.组织任务
通过一定的数据结构把一些任务给放到一起,标准库有一个专门的数据结构PriorityQueue,这里用到的数据结构是PriorityBlockingQueue,及带有优先级又带有阻塞队列。此处的队列要考虑到线程安全问题,可能在多个线程里进行注册任务,同时还有一个专门的线程来取任务执行,此时的队列就需要注意线程安全问题。
其次为了避免盲等的现象,可以使用wait这样的机制指定等待时间时间到了自然唤醒,计算出当前时间和任务目标的时间差即可。既然是指定一个等待时间为啥不用sleep而是用wait,sleep不能被中途唤醒,wait能够被中途唤醒。在等待过程中可能有新的任务插入,新的任务是可能出现在之前所有任务的最前面,在schedule操作中就需要加上一个notify操作。
3.执行时间到了的任务
需要执行时间最靠前的任务,就需要一个线程不停地去检查当前优先队列的对手元素,看看当前最靠前的任务是不是时间到了
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; } public void run(){ runnable.run(); } public long getTime(){ return time; } @Override public int compareTo(MyTask o) { //小的在前 return (int)(this.time - o.time); } } class MyTimer{ private Object locker = new Object(); //定时器内不要能够存放多个任务 private PriorityBlockingQueue<MyTask> queue = new PriorityBlockingQueue<>(); public void schedule(Runnable runnable, long delay){ MyTask task = new MyTask(runnable, delay); queue.put(task); //每次任务插入成功之后都唤醒一下扫描线程,让线程重新检查队首的任务是否时间到要执行 synchronized (locker){ locker.notify(); } } public MyTimer(){ Thread t = new Thread(() ->{ while(true){ try{ //先取出队首元素 MyTask task = queue.take(); //在比较一下看看当前这个任务时间到了没 long curTime = System.currentTimeMillis(); if(curTime < task.getTime()){ //时间没到把任务赛回到队列中 queue.put(task); //制定一个等待时间 synchronized (locker){ locker.wait(task.getTime() - curTime); } }else{ //时间到了执行任务 task.run(); } } catch (InterruptedException e) { e.printStackTrace(); } } }); t.start(); } } public class 定时器 { public static void main(String[] args) { MyTimer myTimer = new MyTimer(); myTimer.schedule(new Runnable() { @Override public void run() { System.out.println("hello timer"); } }, 3000); myTimer.schedule(new Runnable() { @Override public void run() { System.out.println("hello "); } }, 2000); } }
加载全部内容