corn表达式是:由若干数字、空格、符号按一定的规则,组成的一组字符串,从而表达时间的信息。
好像和正则表达式有点类似哈,都是一个字符串表示一些信息。
Cron 表达式生成器: https://www.smart-tools.cn/cron
Cron 表达式是一个具有时间含义的字符串,字符串以 5 或 6 个空格隔开,分为 6 或 7 个域,每一个域代表一种含义。 Cron 有如下两种语法格式:
秒 分 小时 日期 月份 星期
秒 分 小时 日期 月份 星期 年
即:秒 分 小时 日期 月份 星期 年(可为空)
目前的 Cron 表达式主要有两类,分别是:
Linux crontab 命令 (Crontab 是linux系统自带的定时任务,用于设置周期性执行的本地脚本。Crontab的cron表达式只能精确到分钟。例如,* * * * ?)
Java Quartz(Quartz 是一个完全由 Java 编写的开源作业调度框架,为 Java 应用进行任务调度提供了简单却强大的机制。Quartz的cron表达式可以精确到秒。例如,* * * * * ?)
其中,Linux crontab 仅支持分钟级别的任务调度;Java Quartz 则可以秒级别的任务调度;
Linux crontab 中的 cron 语法规范
* * * * *
- - - - -
| | | | |
| | | | +----- 星期中星期几 (0 - 6) (星期天为0)
| | | +---------- 月份 (1 - 12)
| | +--------------- 一个月中的第几天 (1 - 31)
| +-------------------- 小时 (0 - 23)
+------------------------- 分钟 (0 - 59)
Java Quartz 中的 cron 语法规范
* * * * * *
- - - - - -
| | | | | |
| | | | | +----- 星期中星期几 (0 - 6) (星期天为0)
| | | | +---------- 月份 (1 - 12)
| | | +--------------- 一个月中的第几天 (1 - 31)
| | +-------------------- 小时 (0 - 23)
| +------------------------- 分钟 (0 - 59)
+------------------------------ 秒 (0 - 60)
下表为 Cron 表达式中每个域能够取的值以及支持的特殊字符。
域 |
是否必需 |
取值范围 |
特殊字符 |
秒 |
是 |
[0, 59] |
* , - / |
分钟 |
是 |
[0, 59] |
* , - / |
小时 |
是 |
[0, 23] |
* , - / |
日期 |
是 |
[1, 31] |
* , - / ? L W C |
月份 |
是 |
[1, 12] |
* , - / |
星期 |
是 |
[1, 7] 其中 1 表示星期一,7 表示星期日。 |
* , - / ? L C # |
年 |
否 |
留空,1970~2099 |
, - * / |
Cron 表达式中的每个域都支持一定数量的特殊字符,每个特殊字符有其特殊含义。
特殊字符 |
含义 |
示例 |
* |
所有可能的值 |
在月份域中,* 表示每个月;在星期域中,* 表示星期的每一天。 |
, |
列出枚举值 |
在分钟域中,5,20 表示分别在 5 分钟和 20 分钟触发一次。 |
- |
范围 |
在分钟域中,5-20 表示从 5 分钟到 20 分钟之间每隔一分钟触发一次。 |
/ |
表示起始时间开始触发,然后每隔固定时间触发一次 |
在分钟域中,0/15 表示从第 0 分钟开始,每 15 分钟触发一次。在分钟域中3/20表示从第 3 分钟开始,每 20 分钟触发一次。 |
? |
不指定值,仅日期和星期域支持该字符 |
当日期或星期域其中之一被指定了值以后,为了避免冲突,需要将另一个域的值设为?。 |
L |
单词 Last 的首字母,表示最后一天,仅日期和星期域支持该字符 |
在日期域中,L 表示某个月的最后一天。在星期域中,L 表示一个星期的最后一天,也就是星期日(SUN)。如果在 L 前有具体的内容,例如,在星期域中的 6L,表示这个月的最后一个星期六。 |
W |
表示有效工作日(周一到周五),只能出现在日期域,系统将在离指定日期最近的有效工作日触发事件。W 字符寻找当前月份中最近有效工作日,连用字符 LW 时表示为指定月份的最后一个工作日。 |
在日期域中使用 5W, 如果 5 号是星期六,则将在最近的工作日星期五,即 4 日触发。如果 5 日是星期天,则将在最近的工作日星期一,即 6 日触发;如果 5 日在星期一到星期五中的一天,则就在 5 日触发。 |
# |
确定每个月第几个星期几,仅星期域支持该字符。 |
在星期域中,4#2表示某月的第二个星期四。 |
C |
这个字符依靠一个指定的“日历”。也就是说这个表达式的值依赖于相关的“日历”的计算结果,如果没有“日历”关联,则等价于所有包含的“日历”。仅日期和星期域支持该字符。 |
日期域是“5C”表示关联“日历”中第一天,或者这个月开始的第一天的后5天。星期域是“1C”表示关联“日历”中第一天,或者星期的第一天的后1天,也就是周日的后一天(周一)。 |
允许值范围: 0~59 ,不允许为空值,若值不合法,调度器将抛出SchedulerException异常
"*" 代表每隔1秒钟触发
"," 代表在指定的秒数触发,比如"0,15,45"代表0秒、15秒和45秒时触发任务
"-" 代表在指定的范围内触发,比如"25-45"代表从25秒开始触发到45秒结束触发,每隔1秒触发1次
"/" 代表触发步进(step),"/"前面的值代表初始值("*"等同"0"),后面的值代表偏移量,比如"0/20"或者"*/20"代表从0秒钟开始,每隔20秒钟触发1次,即0秒触发1次,20秒触发1次,40秒触发1次;"5/20"代表5秒触发1次,25秒触发1次,45秒触发1次;"10-45/20"代表在[10,45]内步进20秒命中的时间点触发,即10秒触发1次,30秒触发1次
允许值范围: 0~59 ,不允许为空值,若值不合法,调度器将抛出SchedulerException异常
"*" 代表每隔1分钟触发
"," 代表在指定的分钟触发,比如"10,20,40"代表10分钟、20分钟和40分钟时触发任务
"-" 代表在指定的范围内触发,比如"5-30"代表从5分钟开始触发到30分钟结束触 发,每隔1分钟触发
"/" 代表触发步进(step),"/"前面的值代表初始值("*"等同"0"),后面的值代表偏移量,比如"0/25"或者"*/25"代表从0分钟开始,每隔25分钟触发1次,即0分钟触发1次,第25分钟触发1次,第50分钟触发1次;"5/25"代表5分钟触发1次,30分钟触发1次,55分钟触发1次;"10-45/20"代表在[10,45]内步进20分钟命中的时间点触发,即10分钟触发1次,30分钟触发1次
允许值范围: 0~23 ,不允许为空值,若值不合法,调度器将抛出SchedulerException异常
"*" 代表每隔1小时触发
"," 代表在指定的时间点触发,比如"10,20,23"代表10点钟、20点钟和23点触发任务
"-" 代表在指定的时间段内触发,比如"20-23"代表从20点开始触发到23点结束触发,每隔1小时触发
"/" 代表触发步进(step),"/"前面的值代表初始值("*"等同"0"),后面的值代表偏移量,比如"0/1"或者"*/1"代表从0点开始触发,每隔1小时触发1次;"1/2"代表从1点开始触发,以后每隔2小时触发一次
允许值范围: 1~12 (JAN-DEC),不允许为空值,若值不合法,调度器将抛出SchedulerException异常
"*" 代表每个月都触发
"," 代表在指定的月份触发,比如"1,6,12"代表1月份、6月份和12月份触发任务
"-" 代表在指定的月份范围内触发,比如"1-6"代表从1月份开始触发到6月份结束触发,每隔1个月触发
"/" 代表触发步进(step),"/"前面的值代表初始值("*"等同"1"),后面的值代表偏移量,比如"1/2"或者"*/2"代表从1月份开始触发,每隔2个月触发1次;"6/6"代表从6月份开始触发,以后每隔6个月触发一次;"1-6/12"表达式意味着每年1月份触发
"C" 这个字符依靠一个指定的“日历”。也就是说这个表达式的值依赖于相关的“日历”的计算结果,如果没有“日历”关联,则等价于所有包含的“日历”。比如“5C”表示关联“日历”中第一天,或者这个月开始的第一天的后5天。
允许值范围: 1~7 (SUN-SAT),1代表星期天(一星期的第一天),以此类推,7代表星期六(一星期的最后一天),不允许为空值,若值不合法,调度器将抛出SchedulerException异常
"*" 代表每星期都触发;
"?" 与{日期}互斥,即意味着若明确指定{日期}触发,则表示{星期}无意义,以免引起冲突和混乱
"," 代表在指定的星期约定触发,比如"1,3,5"代表星期天、星期二和星期四触发
"-" 代表在指定的星期范围内触发,比如"2-4"代表从星期一开始触发到星期三结束触发,每隔1天触发
"/" 代表触发步进(step),"/"前面的值代表初始值("*"等同"1"),后面的值代表偏移量,比如"1/3"或者"*/3"代表从星期天开始触发,每隔3天触发1次;"1-5/2"表达式意味着在[1,5]范围内,每隔2天触发,即星期天、星期二、星期四触发
"L" 如果{星期}占位符如果是"L",即意味着星期的的最后一天触发,即星期六触发,L= 7或者 L = SAT,因此,"5L"意味着一个月的最后一个星期四触发
"#" 用来指定具体的周数,"#"前面代表星期,"#"后面代表本月第几周,比如"2#2"表示本月第二周的星期一,"5#3"表示本月第三周的星期四,因此,"5L"这种形式只不过是"#"的特殊形式而已
"C" 这个字符依靠一个指定的“日历”。也就是说这个表达式的值依赖于相关的“日历”的计算结果,如果没有“日历”关联,则等价于所有包含的“日历”。比如:“1C”表示关联“日历”中第一天,或者星期的第一天的后1天,也就是周日的后一天(周一)。
允许值范围: 1970~2099 ,允许为空,若值不合法,调度器将抛出SchedulerException异常
"*"代表每年都触发
","代表在指定的年份才触发,比如"2011,2012,2013"代表2011年、2012年和2013年触发任务
"-"代表在指定的年份范围内触发,比如"2011-2020"代表从2011年开始触发到2020年结束触发,每隔1年触发
"/"代表触发步进(step),"/"前面的值代表初始值("*"等同"1970"),后面的值代表偏移量,比如"2011/2"或者"*/2"代表从2011年开始触发,每隔2年触发1次
注意:除了{日期}和{星期}可以使用"?"来实现互斥,表达无意义的信息之外,其他占位符都要具有具体的时间含义,且依赖关系为:年->月->日期(星期)->小时->分钟->秒数
示例 1 |
说明 |
0 15 10 ? * * |
每天上午 10:15 执行任务 |
0 15 10 * * ? |
每天上午 10:15 执行任务 |
0 0 12 * * ? |
每天中午 12:00 执行任务 |
0 0 10,14,16 * * ? |
每天上午 10:00 点、下午 14:00 以及下午 16:00 执行任务 |
0 0/30 9-17 * * ? |
每天上午 09:00 到下午 17:00 时间段内每隔半小时执行任务 |
0 * 14 * * ? |
每天下午 14:00 到下午 14:59 时间段内每隔 1 分钟执行任务 |
0 0-5 14 * * ? |
每天下午 14:00 到下午 14:05 时间段内每隔 1 分钟执行任务 |
0 0/5 14 * * ? |
每天下午 14:00 到下午 14:55 时间段内每隔 5 分钟执行任务 |
0 0/5 14,18 * * ? |
每天下午 14:00 到下午 14:55、下午 18:00 到下午 18:55 时间段内每隔 5 分钟执行任务 |
0 0 12 ? * WED |
每个星期三中午 12:00 执行任务 |
0 15 10 15 * ? |
每月 15 日上午 10:15 执行任务 |
0 15 10 L * ? |
每月最后一日上午 10:15 执行任务 |
0 15 10 ? * 6L |
每月最后一个星期六上午 10:15 执行任务 |
0 15 10 ? * 6#3 |
每月第三个星期六上午 10:15 执行任务 |
0 10,44 14 ? 3 WED |
每年 3 月的每个星期三下午 14:10 和 14:44 执行任务 |
cron 表达式最主要的就是在程序中做一些定时任务,比如某些系统的报表数据,某些游戏的排行榜,由于这些数据量实时统计非常消耗程序性能,所以就每隔一段时间,通过自动任务跑一次,这样可以极大的提升用户浏览体验,要是在游戏里,还可以增加一种神秘感。
另外,某些具体点的数据拉取,比如你如果从事平台对接工作,要从某些平台下载你的订单,那么肯定是每隔多久抓一次。
又比如你写个爬虫,要实时的了解你的某些数据,然后从这些数据中反应你的情况。
org.quartz-scheduler
quartz
2.2.3
定义一个HelloJob类实现Job接口
import org.quartz.Job;
import org.quartz.JobExecutionContext;
import org.quartz.JobExecutionException;
import java.util.Date;
public class HelloJob implements Job {
@Override
public void execute(JobExecutionContext arg0) throws JobExecutionException {
System.out.println("Say hello to Quartz {" + new Date() +"}");
}
}
写个主函数,在主函数里面完成整个Quartz的操作过程
import org.quartz.*;
import org.quartz.impl.StdSchedulerFactory;
import java.util.Date;
public class HelloQuartz {
public static void main(String[] args) throws SchedulerException {
//1.创建Scheduler的工厂
SchedulerFactory sf = new StdSchedulerFactory();
//2.从工厂中获取调度器实例
Scheduler scheduler = sf.getScheduler();
//3.创建JobDetail
JobDetail jb = JobBuilder.newJob(HelloJob.class)
.withDescription("this is a job") //job的描述
.withIdentity("Job", "Group") //job 的name和group
.build();
//任务运行的时间,SimpleSchedle类型触发器有效
long time= System.currentTimeMillis() + 3*1000L; //3秒后启动任务
Date statTime = new Date(time);
//4.创建Trigger
//使用SimpleScheduleBuilder或者CronScheduleBuilder
Trigger t = TriggerBuilder.newTrigger()
.withDescription("")
.withIdentity("Trigger", "TriggerGroup")
//.withSchedule(SimpleScheduleBuilder.simpleSchedule())
.startAt(statTime) //默认当前时间启动
.withSchedule(CronScheduleBuilder.cronSchedule("0/10 * * * * ?")) // 10秒执行一次
.build();
//5.注册任务和定时器
scheduler.scheduleJob(jb, t);
//6.启动 调度器
scheduler.start();
}
}
org.springframework.boot
spring-boot-starter-quartz
定义一个HelloJob类实现Job接口
package com.example.demo;
import org.quartz.Job;
import org.quartz.JobExecutionContext;
import org.quartz.JobExecutionException;
import java.util.Date;
public class HelloJob implements Job {
@Override
public void execute(JobExecutionContext arg0) throws JobExecutionException {
System.out.println("Say hello to Quartz {" + new Date() +"}");
}
}
写个主函数,在主函数里面完成整个Quartz的操作过程
package com.example.demo;
import org.quartz.*;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
// 要配置Quartz的调度器Scheduler
// 调度器由SpringBoot管理,所以就变成了配置Spring
@Configuration
public class Demo {
// 配置的核心是向Spring容器保存一个job和保存一个Trigger
// 创建一个封装Job对象的类型JobDetail
// 使用@Bean注解标记的方法将这个对象保存到Spring容器
@Bean
public JobDetail addStock(){
//newJob方法就是在绑定要运行的Job接口实现类,需要实现类的反射做参数
return JobBuilder.newJob(HelloJob.class)
// 给当前JobDetail对象在调度环境中起名
.withIdentity("addStock")
// 即使没有触发器绑定当前JobDetail对象,也不会被删除
.storeDurably()
.build();
}
// 下面是触发器的声明,也会保存到Spring容器中
// 它能够设置job的运行时机
@Bean
public Trigger addStockTrigger(){
System.out.println("Trigger保存到Spring容器中");
// 定义Cron表达式
CronScheduleBuilder cron=
CronScheduleBuilder.cronSchedule("0/10 * * * * ?"); // 10秒执行一次
return TriggerBuilder.newTrigger()
// 绑定要运行的JobDetail对象
.forJob(addStock())
// 为触发器起名
.withIdentity("addStockTrigger")
// 绑定cron表达式
.withSchedule(cron)
.build();
}
}
在Linux中的cron表达式与上文所述有所不同,更准确地说是更加精简了。
crontab的命令构成为 cron表示式 + command,这里的cron表达
Minutes Hours DayOfMonth Month DayOfWeek
而操作符则有:
* / - ,
一起来看看几个例子:
30 21 * * * /etc/init.d/smb restart
0 23 * * 6 /etc/init.d/smb restart
0 23-7/1 * * * /etc/init.d/smb restart