public class TimerTaskApp {
public static void main(String[] args) {
// 创建一个定时器对象
Timer timer = new Timer();
// task是一个具体要完成的定时任务对象(即要做什么)
TimerTask task = new TimerTask() {
@Override
public void run() { //多线程
//这里定义你定时要做的事情
System.out.println("timer task run...");
}
};
// schedule(定时任务对象,定时任务的开始时间,多久运行一次)
timer.schedule(task,0,2000);
}
}
简单说就是你定时干什么事情,这就是工作,工作不可能就是一个简单的方法,还要设置一些明细信息。工作啥时候执行,设置一个调度器,可以简单理解成设置一个工作执行的时间。工作和调度都是独立定义的,它们两个怎么配合到一起呢?用触发器
格式:{秒数} {分钟} {小时} {日期} {月份} {星期} {年份(可为空,spring不可以用此字段)}
(“0/5 * * * * ?”):表示从每分钟的0秒开始,每隔5秒触发一次。
cron 中的通配符的认识:
(1)* :表示匹配该域的任意值。(* * * * * ?) ,每分每秒都在触发
(2)? :表示不指定值,只能用在[日]和[周]两个域。一般指定了日期就不用指定周了,或者指定了周几就不用指定日期:(“1 * * * * ?“):表示每分钟的第一秒触发
(3)- :表示该域的连续范围。(“5-10 * * * * ?”):表示在每分钟的5秒到10秒之间,每秒触发一次。
(4)/ :该符号将所在域中的表达式分为两个部分,第一部分表示起始时间,第一部分表示间隔时间。(“0/1 * * * * ?”):表示从每分钟的第0秒开始,每隔1秒执行一次。这里的0就表示起始时间,1就表示间隔时间
(5), :表示匹配该域的指定值。(”* 1,2,3 * * * ?")则表示分别在第1 分、第 2分、 第3分的每秒时间都执行该定时任务。
(6)L :表示表示英文中的last 的意思,只能出现在[日] 和 [周]域。如果在”L”前加上数字,则表示该数据的最后一个。例如在[周]域使用5L,意味着在最后的一个星期四触发。 ([周]域里面1表示周日,2表示周一,以此类推,7表示周六)
(7)W :表示有效工作日(周一到周五),只能出现在[日]域且只能用在具体的数字之后,系统将在离指定日期最近的有效工作日触发定时任务。例如在[日]域使用5W,如果5号在星期一到星期五中的一天,则就在5号触发;如果5号是星期六,则将在最近的工作日:星期五,即4号触发;如果5号是星期天,则在6号(周一)触发。另外一点,W的最近寻找不会跨过月份 。例如 1号是星期六,“1W” 就只能往本月的下一个最近的工作日推不能跨月往上一个月推。
(8)LW :这两个字符可以连用,表示在某个月最后一个工作日,即最后一个星期五。
(9)# :用于确定每个月第几个周几,只能出现在[周]域。例如”2#3” 表示在每月的第三个周一。([周]域里面1表示周日,2表示周一,以此类推,7表示周六)
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-quartz</artifactId>
</dependency>
public class MyQuartz extends QuartzJobBean {
@Override
protected void executeInternal(JobExecutionContext context) throws JobExecutionException {
// context是工作执行的上下文对象
// 方法里面定义具体要做的事情
System.out.println("quartz task run...");
}
}
//配置类
@Configuration
public class QuartzConfig {
// 工作明细:
// 工作明细中要设置对应的具体工作,使用newJob()操作传入对应的工作任务类型即可。
@Bean
public JobDetail printJobDetail(){
//绑定具体的工作(具体要做什么),
// MyQuartz.class:定义指定的工作种类
// storeDurably()用来做持久化的,如果没有使用,会帮你自动存储
return JobBuilder.newJob(MyQuartz.class).storeDurably().build();
}
// 触发器
@Bean
public Trigger printJobTrigger(){
// 使用cron表达式,用来描述任务的执行时间(秒 分 时 日 月 星期)
ScheduleBuilder schedBuilder = CronScheduleBuilder.cronSchedule("0/5 * * * * ?");
//绑定对应的工作明细
return TriggerBuilder.newTrigger().forJob(printJobDetail()).withSchedule(schedBuilder).build();
}
}
触发器需要绑定任务,使用forJob()操作传入绑定的工作明细对象。此处可以为工作明细设置名称然后使用名称绑定,也可以直接调用对应方法绑定。触发器中最核心的规则是执行时间,此处使用调度器定义执行时间,执行时间描述方式使用的是cron表达式。有关cron表达式的规则,各位小伙伴可以去参看相关课程学习,略微复杂,而且格式不能乱设置,不是写个格式就能用的,写不好就会出现冲突问题。
spring根据定时任务的特征,将定时任务的开发简化到了极致。
步骤①:开启定时任务功能,在引导类上开启定时任务功能的开关,使用注解@EnableScheduling
@SpringBootApplication
//开启定时任务功能
@EnableScheduling
public class TaskApplication {
public static void main(String[] args) {
SpringApplication.run(TaskApplication.class,args);
}
}
@Component
public class MyBean {
@Scheduled(cron = "0/1 * * * * ?")
public void print(){
System.out.println(Thread.currentThread().getName()+" :spring task run...");
}
}
这就完成了定时任务的配置。总体感觉其实什么东西都没少,只不过没有将所有的信息都抽取成bean,而是直接使用注解绑定定时执行任务的事情而已。
spring:
task:
scheduling:
pool:
size: 1 # 任务调度线程池大小 默认 1
thread-name-prefix: ssm_ # 调度线程名称前缀 默认 scheduling-
shutdown:
await-termination: false # 线程池关闭时 是否 等待所有任务完成
await-termination-period: 10s # 调度线程关闭前 最大等待时间,确保最后一定关闭