在 Spring + SpringMVC 環境中,通常來講,要實現定時任務,咱們有兩中方案,一種是使用 Spring 自帶的定時任務處理器 @Scheduled 註解,另外一種就是使用第三方框架 Quartz ,Spring Boot 源自 Spring+SpringMVC ,所以自然具有這兩個 Spring 中的定時任務實現策略,固然也支持 Quartz,本文咱們就來看下 Spring Boot 中兩種定時任務的實現方式。java
使用 @Scheduled 很是容易,直接建立一個 Spring Boot 項目,而且添加 web 依賴 spring-boot-starter-web
,項目建立成功後,添加 @EnableScheduling
註解,開啓定時任務:git
@SpringBootApplication
@EnableScheduling
public class ScheduledApplication {
public static void main(String[] args) {
SpringApplication.run(ScheduledApplication.class, args);
}
}
複製代碼
接下來配置定時任務:github
@Scheduled(fixedRate = 2000)
public void fixedRate() {
System.out.println("fixedRate>>>"+new Date());
}
@Scheduled(fixedDelay = 2000)
public void fixedDelay() {
System.out.println("fixedDelay>>>"+new Date());
}
@Scheduled(initialDelay = 2000,fixedDelay = 2000)
public void initialDelay() {
System.out.println("initialDelay>>>"+new Date());
}
複製代碼
上面這是一個基本用法,除了這幾個基本屬性以外,@Scheduled 註解也支持 cron 表達式,使用 cron 表達式,能夠很是豐富的描述定時任務的時間。cron 表達式格式以下:web
[秒] [分] [小時] [日] [月] [周] [年]spring
具體取值以下:後端
序號 | 說明 | 是否必填 | 容許填寫的值 | 容許的通配符 |
---|---|---|---|---|
1 | 秒 | 是 | 0-59 | - * / |
2 | 分 | 是 | 0-59 | - * / |
3 | 時 | 是 | 0-23 | - * / |
4 | 日 | 是 | 1-31 | - * ? / L W |
5 | 月 | 是 | 1-12 or JAN-DEC | - * / |
6 | 周 | 是 | 1-7 or SUN-SAT | - * ? / L ## |
7 | 年 | 否 | 1970-2099 | - * / |
這一塊須要你們注意的是,月份中的日期和星期可能會起衝突,所以在配置時這兩個得有一個是 ?
框架
通配符含義:前後端分離
?
表示不指定值,即不關心某個字段的取值時使用。須要注意的是,月份中的日期和星期可能會起衝突,所以在配置時這兩個得有一個是 ?
*
表示全部值,例如:在秒的字段上設置 *
,表示每一秒都會觸發,
用來分開多個值,例如在周字段上設置 "MON,WED,FRI" 表示週一,週三和週五觸發-
表示區間,例如在秒上設置 "10-12",表示 10,11,12秒都會觸發/
用於遞增觸發,如在秒上面設置"5/15" 表示從5秒開始,每增15秒觸發(5,20,35,50)##
序號(表示每個月的第幾個周幾),例如在周字段上設置"6##3"表示在每個月的第三個週六,(用 在母親節和父親節再合適不過了)L
表示最後的意思。在日字段設置上,表示當月的最後一天(依據當前月份,若是是二月還會自動判斷是不是潤年), 在周字段上表示星期六,至關於"7"或"SAT"(注意週日算是第一天)。若是在"L"前加上數字,則表示該數據的最後一個。例如在周字段上設置"6L"這樣的格式,則表示"本月最後一個星期五"W
表示離指定日期的最近工做日(週一至週五),例如在日字段上設置"15W",表示離每個月15號最近的那個工做日觸發。若是15號正好是週六,則找最近的週五(14號)觸發, 若是15號是周未,則找最近的下週一(16號)觸發,若是15號正好在工做日(週一至週五),則就在該天觸發。若是指定格式爲 "1W",它則表示每個月1號日後最近的工做日觸發。若是1號正是週六,則將在3號下週一觸發。(注,"W"前只能設置具體的數字,不容許區間"-")L
和 W
能夠一組合使用。若是在日字段上設置"LW",則表示在本月的最後一個工做日觸發(通常指發工資 )例如,在 @Scheduled 註解中來一個簡單的 cron 表達式,每隔5秒觸發一次,以下:ide
@Scheduled(cron = "0/5 * * * * *")
public void cron() {
System.out.println(new Date());
}
複製代碼
上面介紹的是使用 @Scheduled 註解的方式來實現定時任務,接下來咱們再來看看如何使用 Quartz 實現定時任務。spring-boot
通常在項目中,除非定時任務涉及到的業務實在是太簡單,使用 @Scheduled 註解來解決定時任務,不然大部分狀況可能都是使用 Quartz 來作定時任務。在 Spring Boot 中使用 Quartz ,只須要在建立項目時,添加 Quartz 依賴便可:
項目建立完成後,也須要添加開啓定時任務的註解:
@SpringBootApplication
@EnableScheduling
public class QuartzApplication {
public static void main(String[] args) {
SpringApplication.run(QuartzApplication.class, args);
}
}
複製代碼
Quartz 在使用過程當中,有兩個關鍵概念,一個是JobDetail(要作的事情),另外一個是觸發器(何時作),要定義 JobDetail,須要先定義 Job,Job 的定義有兩種方式:
第一種方式,直接定義一個Bean:
@Component
public class MyJob1 {
public void sayHello() {
System.out.println("MyJob1>>>"+new Date());
}
}
複製代碼
關於這種定義方式說兩點:
第二種定義方式,則是繼承 QuartzJobBean 並實現默認的方法:
public class MyJob2 extends QuartzJobBean {
HelloService helloService;
public HelloService getHelloService() {
return helloService;
}
public void setHelloService(HelloService helloService) {
this.helloService = helloService;
}
@Override
protected void executeInternal(JobExecutionContext jobExecutionContext) throws JobExecutionException {
helloService.sayHello();
}
}
public class HelloService {
public void sayHello() {
System.out.println("hello service >>>"+new Date());
}
}
複製代碼
和第1種方式相比,這種方式支持傳參,任務啓動時,executeInternal 方法將會被執行。
Job 有了以後,接下來建立類,配置 JobDetail 和 Trigger 觸發器,以下:
@Configuration
public class QuartzConfig {
@Bean
MethodInvokingJobDetailFactoryBean methodInvokingJobDetailFactoryBean() {
MethodInvokingJobDetailFactoryBean bean = new MethodInvokingJobDetailFactoryBean();
bean.setTargetBeanName("myJob1");
bean.setTargetMethod("sayHello");
return bean;
}
@Bean
JobDetailFactoryBean jobDetailFactoryBean() {
JobDetailFactoryBean bean = new JobDetailFactoryBean();
bean.setJobClass(MyJob2.class);
JobDataMap map = new JobDataMap();
map.put("helloService", helloService());
bean.setJobDataMap(map);
return bean;
}
@Bean
SimpleTriggerFactoryBean simpleTriggerFactoryBean() {
SimpleTriggerFactoryBean bean = new SimpleTriggerFactoryBean();
bean.setStartTime(new Date());
bean.setRepeatCount(5);
bean.setJobDetail(methodInvokingJobDetailFactoryBean().getObject());
bean.setRepeatInterval(3000);
return bean;
}
@Bean
CronTriggerFactoryBean cronTrigger() {
CronTriggerFactoryBean bean = new CronTriggerFactoryBean();
bean.setCronExpression("0/10 * * * * ?");
bean.setJobDetail(jobDetailFactoryBean().getObject());
return bean;
}
@Bean
SchedulerFactoryBean schedulerFactoryBean() {
SchedulerFactoryBean bean = new SchedulerFactoryBean();
bean.setTriggers(cronTrigger().getObject(), simpleTriggerFactoryBean().getObject());
return bean;
}
@Bean
HelloService helloService() {
return new HelloService();
}
}
複製代碼
關於這個配置說以下幾點:
所有定義完成後,啓動 Spring Boot 項目就能夠看到定時任務的執行了。
這裏主要向你們展現了 Spring Boot 中整合兩種定時任務的方法,整合成功以後,剩下的用法基本上就和在 SSM 中使用一致了,再也不贅述。本文案例我已經上傳到 GitHub ,歡迎你們 star:github.com/lenve/javab…
關注公衆號【江南一點雨】,專一於 Spring Boot+微服務以及先後端分離等全棧技術,按期視頻教程分享,關注後回覆 Java ,領取鬆哥爲你精心準備的 Java 乾貨!