1、定时任务
按照指定时间执行的程序。
使用场景
- 数据分析
- 数据清理
- 系统服务监控
2、同步和异步
同步调用
- 程序按照代码顺序依次执行,每一行程序都必须等待上一行程序执行完成之后才能执行;
异步调用 (异步就是线程与线程不等待)
- 顺序执行时,不等待异步调用的代码块返回结果就执行后面的程序。
App.java
@SpringBootApplication
@EnableScheduling // 启动定时器
@EnableAsync // 支持异步定时器
public class App {
public static void main(String[] args) {
SpringApplication.run(App.class, args);
}
}
application.xml
spring:
task:
# Spring 执行器配置,对应 TaskExecutionProperties 配置类。对于 Spring 异步任务,会使用该执行器。
execution:
thread-name-prefix: async- # 线程池的线程名的前缀。默认为 task- ,建议根据自己应用来设置
pool: # 线程池相关
core-size: 8 # 核心线程数,线程池创建时候初始化的线程数。默认为 8 。
max-size: 20 # 最大线程数,线程池最大的线程数,只有在缓冲队列满了之后,才会申请超过核心线程数的线程。默认为 Integer.MAX_VALUE
keep-alive: 60s # 允许线程的空闲时间,当超过了核心线程之外的线程,在空闲时间到达之后会被销毁。默认为 60 秒
queue-capacity: 200 # 缓冲队列大小,用来缓冲执行任务的队列的大小。默认为 Integer.MAX_VALUE 。
allow-core-thread-timeout: true # 是否允许核心线程超时,即开启线程池的动态增长和缩小。默认为 true 。
shutdown:
await-termination: true # 应用关闭时,是否等待定时任务执行完成。默认为 false ,建议设置为 true
await-termination-period: 60 # 等待任务完成的最大时长,单位为秒。默认为 0 ,根据自己应用来设置
ScheduleTask.java
@Slf4j
@Component
public class ScheduleTask {
@Scheduled(cron = "0/1 * * * * *")
public void synTask() {
sleep();
System.out.println(Thread.currentThread().getName() + " syn-task 执行,当前时间: " + LocalDateTime.now());
}
@Async
@Scheduled(cron = "0/1 * * * * *")
public void asyncTask() {
sleep();
System.out.println(Thread.currentThread().getName() + " async-task 执行,当前时间: " + LocalDateTime.now());
}
private void sleep() {
try {
Thread.sleep(10 * 1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
定时器执行规则注解
@Scheduled(fixedRate = 5000) :上一次开始执行时间点之后5秒再执行
@Scheduled(fixedDelay = 5000) :上一次执行完毕时间点之后5秒再执行
@Scheduled(initialDelay=1000, fixedRate=5000) :第一次延迟1秒后执行,之后按fixedRate的规则每5秒执行一次
@Scheduled(cron="/5") :通过cron表达式定义规则
cron表达式详解
cron表达式有至少6个(也可能7个)有空格分隔的时间元素。
格式: cron = [ 秒 ] [ 分钟 ] [ 小时 ] [ 日 ] [ 月 ] [ 周(星期)] [ 年份 ]
序号 | 元素 | 值 | 通配符 |
---|---|---|---|
1 | 秒 | 0-59 | - * / |
2 | 分钟 | 0-59 | - * / |
3 | 小时 | 0-23 | - * / |
4 | 日 | 1-31 | - * ? / L W |
5 | 月 | 1-12 or JAN-DEC | ,- * / |
6 | 周(星期) | 1~7 1=SUN 或 SUN,MON,TUE,WED,THU,FRI,SAT | ,- * ? / L # |
7 | 年份 | 1970-2099 | ,- * / |
通配符详解
序号 | 通配符 | 备注 |
---|---|---|
1 | * | 表示所有值,如:秒的位置上是 * ,则代表每一秒都会执行 |
2 | ? | 表示不指定值,如:年的位置上是 ?,则代表不知道某一年 |
3 | - | 表示区间,如:小时的位是1-2,代表小时为1,2都会执行 |
4 | , | 表示指定多个值,如:小时为1,2,3时都会运行 |
5 | L | 表示最后的意思,如:分钟是L代表每一小时最后一分钟都会运行 |
6 | W | 表示离指定日期的最近那个工作日,如:10W代表离10号最近的一个工作日运行 |
7 | # | 表示第几个,如:2#3 代表二月的第三周执行 |
下面是常见的 corn 表达式:
表达式 | 备注 |
---|---|
*/5 * * * * ? | 每隔5秒执行一次 |
0 */1 * * * ? | 每隔1分钟执行一次 |
0 0 23 * * ? | 每天23点执行一次 |
0 0 1 1 * ? | 每月1号凌晨1点执行一次 |
0 0 23 L * ? | 每月最后一天23点执行一次 |
0 0 1 ? * L | 每周星期天凌晨1点执行一次 |
0 15,25,35 * * * ? | 在15分、25分、35分执行一次 |
0 * 16 * * ? | 每天下午16点钟开始到16点59分结束这么一个时间段 |
0 0/5 16 * * ? | 每天下午16点到16点55分之间每5分钟触发一次 |
0 0/5 14,18 * * ? | 每天下午14点到14点55分 和18点到18点55分之间 每5分钟触发一次 |
0 0-5 14 * * ? | 每天下午14点开始到14点05结束 |
0 10,44 14 ? 3 WED | 每年三月份的每个周三下午14点10分和14点44各一次 |
0 15 10 ? * MON-FRI | 每个工作日的10点15分 |
0 15 10 15 * ? | 每个月15号的上午10点15分 |
0 15 10 L * ? | 每个月最后一天的10点15分 |
0 15 10 ? * 6L | 每个月最后一个周五的10点15分 |
0 15 10 ? * 6L | 2002-2005 2002年到2005年每个月最后一个周五的10点15分 |
0 15 10 ? * 6#3 | 每个月的第三个周五的10点15分 |
0 0 12 1/5 * ? | 每个月从第一天开始每隔5天中午12点触发一次 |
0 11 11 11 11 ? | 每年11月11号11点11分 |
0 0 0,13,18,21 * * ? | 每天的0点、13点、18点、21点都执行一次 |
0 0/30 9-17 * * ? | 朝九晚五工作时间内每半小时 |
0 0 12 * * ? | 每天中午12点触发 |
0 15 10 ? * * | 每天上午10:15触发 |
0 0 23 * * ? | 每天 |
0 59 23 ? * SUN | 每周 |
0 59 23 L * ? | 每月 |
0 59 23 L 3,6,9,12 ? | 每季度 |
0 0 0 31 12 ? | 每年 |