Java学习笔记:定时任务调度工具之 Timer

举报
彭世瑜 发表于 2021/08/13 23:38:38 2021/08/13
【摘要】 定时任务调度 定义: 基于给定的时间点,给定的时间间隔或者给定的执行次数自动执行的任务 Timer 和 Quartz Timer: 有且仅有一个后台线程对多个业务线程进行定时定频率的调度 主要构件 Timer -定时调用-> TimerTask 1 定时函数的用法 // 延时执行一次 public void schedule(TimerTask...

定时任务调度

定义:

基于给定的时间点,给定的时间间隔或者给定的执行次数自动执行的任务

Timer 和 Quartz

Timer: 有且仅有一个后台线程对多个业务线程进行定时定频率的调度

主要构件

Timer -定时调用-> TimerTask

  
 
  • 1

在这里插入图片描述

定时函数的用法

// 延时执行一次
public void schedule(TimerTask task, long delay)

// 定时执行一次
public void schedule(TimerTask task, Date time)

// 延时间隔执行
public void schedule(TimerTask task, long delay, long period)

// 定时间隔执行
public void schedule(TimerTask task, Date firstTime, long period)

// 延时间隔执行
public void scheduleAtFixedRate(TimerTask task, long delay, long period)

// 定时间隔执行
public void scheduleAtFixedRate(TimerTask task, Date firstTime, long period)

  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17

代码实例

package timer;

import java.text.SimpleDateFormat;
import java.util.Calendar;
import java.util.Timer;
import java.util.TimerTask;

class Util { public static String getCurrentDateTime() { Calendar calendar = Calendar.getInstance(); SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); return dateFormat.format(calendar.getTime()); }
}

class MyTimerTask extends TimerTask { @Override public void run() { System.out.println("MyTimerTask is Running " + Util.getCurrentDateTime()); }
}


public class TimerDemo { public static void main(String[] args) { Timer timer = new Timer(); // 延时0s之后,每隔1s执行一次 timer.schedule(new MyTimerTask(), 0L, 1000L); }
}

  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31

执行结果

MyTimerTask is Running 2020-06-22 22:49:45
MyTimerTask is Running 2020-06-22 22:49:46
MyTimerTask is Running 2020-06-22 22:49:47
...

  
 
  • 1
  • 2
  • 3
  • 4

其他函数

TimerTask.cancel() 取消【当前】 TimerTask 里的任务
TimerTask.scheduledExecutionTime() 返回此任务最近实际执行的已安排执行的时间

Timer.cancel()终止此计时器,丢弃【所有】当前已安排的任务
Timer.purge() 从此计时器的任务队列中移除所有已取消的任务,返回移除数量

  
 
  • 1
  • 2
  • 3
  • 4
  • 5

区别

schedule 和 scheduleAtFixedRate

1、首次计划执行的时间早于当前的时间

schedule fixed-delay

如果第一次执行时间被 delay 了,随后的执行时间按照上一次实际执行【完成的时间点】进行计算

scheduleAtFixedRate fixed-rate

如果第一次执行时间被 delay 了,随后的执行时间按照上一次【开始的时间点】进行计算,并且为了赶上进度会多次执行任务,因此 TimerTask 中的执行体需要考虑同步

代码示例

package timer;

import java.text.SimpleDateFormat;
import java.util.Calendar;
import java.util.Date;
import java.util.Timer;
import java.util.TimerTask;


public class TimerDemo { public static void main(String[] args) { Timer timer = new Timer(); SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); Calendar calendar = Calendar.getInstance(); calendar.add(Calendar.SECOND, -10); timer.schedule(new TimerTask() { @Override public void run() { System.out.println(dateFormat.format(new Date())); } }, calendar.getTime(), 2000L); }
}

  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26

执行结果

timer.schedule

2020-06-22 23:16:56
2020-06-22 23:16:58
2020-06-22 23:17:00


timer.scheduleAtFixedRate
2020-06-22 23:18:19
2020-06-22 23:18:19
2020-06-22 23:18:19

  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11

2、任务执行所需时间超出任务的执行周期间隔
schedule
下次执行时间相对于上一次实际执行【完成的时间点】,因此执行时间会不断延后

scheduleAtFixedRate
下一次执行时间相对于上一次【开始的时间点】,因此执行时间一般不会延后,因此存在并发性

package timer;

import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Timer;
import java.util.TimerTask;


public class TimerDemo { public static void main(String[] args) { Timer timer = new Timer(); SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); timer.scheduleAtFixedRate(new TimerTask() { @Override public void run() { try { Thread.sleep(3000L); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println(dateFormat.format(new Date())); } }, 0L, 1000L); }
}

  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
timer.schedule

2020-06-22 23:24:08
2020-06-22 23:24:11
2020-06-22 23:24:14


timer.scheduleAtFixedRate
2020-06-22 23:25:18
2020-06-22 23:25:21
2020-06-22 23:25:24

  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11

Timer 综合应用

执行 5 次任务,就停止定时器,并退出程序

package timer;

import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Timer;
import java.util.TimerTask;

// 自定义任务
class MyTimerTask extends TimerTask { private SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); private int count = 0; private Timer timer; public MyTimerTask(Timer timer) { this.timer = timer; } @Override public void run() { System.out.println(dateFormat.format(new Date())); // 判断执行满5次就停止 this.count++; if (this.count >= 5) { // 停止当前任务 this.cancel(); // 停止定时器,并退出程序 this.timer.cancel(); } }
}

public class TimerDemo { public static void main(String[] args) { Timer timer = new Timer(); timer.schedule(new MyTimerTask(timer), 0L, 1000L); }
}

  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43

打印结果

2020-06-23 20:46:11
2020-06-23 20:46:12
2020-06-23 20:46:13
2020-06-23 20:46:14
2020-06-23 20:46:15

  
 
  • 1
  • 2
  • 3
  • 4
  • 5

Timer 缺陷

1、管理并发任务的缺陷
Timer 有且仅有一个线程去执行定时任务,如果存在多个任务,且任务时间过长,会导致执行效果与预期不符

2、当任务抛出异常时的缺陷
如果 TimerTask 抛出 RuntimeException,Timer 会停止所有任务的运行

3、以下场景不建议使用

  1. 对时效性要求较高的多任务并发作业
  2. 对复杂的任务的调度

文章来源: pengshiyu.blog.csdn.net,作者:彭世瑜,版权归原作者所有,如需转载,请联系作者。

原文链接:pengshiyu.blog.csdn.net/article/details/106956607

【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。