Spring?Boot使用Schedule實現(xiàn)定時任務的方法
0. 開發(fā)環(huán)境
IDE:IntelliJ IDEA 2017.1 x64
jdk:1.8.0_91
Spring Boot:2.1.1.RELEASE
1. 簡單定時任務
對于一些比較簡單的定時任務,比如固定時間間隔執(zhí)行固定方法,在標準Java方法上注解@Scheduled即可
package cn.wbnull.springbootdemo.schedule;
import cn.wbnull.springbootdemo.util.DateUtils;
import cn.wbnull.springbootdemo.util.LoggerUtils;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;
@Component
public class ScheduledTask {
@Scheduled(cron = "0/10 * * * * ?") //每10秒執(zhí)行一次
public void scheduledTaskByCorn() {
LoggerUtils.info("定時任務開始 ByCorn:" + DateUtils.dateFormat());
scheduledTask();
LoggerUtils.info("定時任務結束 ByCorn:" + DateUtils.dateFormat());
}
@Scheduled(fixedRate = 10000) //每10秒執(zhí)行一次
public void scheduledTaskByFixedRate() {
LoggerUtils.info("定時任務開始 ByFixedRate:" + DateUtils.dateFormat());
scheduledTask();
LoggerUtils.info("定時任務結束 ByFixedRate:" + DateUtils.dateFormat());
}
@Scheduled(fixedDelay = 10000) //每10秒執(zhí)行一次
public void scheduledTaskByFixedDelay() {
LoggerUtils.info("定時任務開始 ByFixedDelay:" + DateUtils.dateFormat());
scheduledTask();
LoggerUtils.info("定時任務結束 ByFixedDelay:" + DateUtils.dateFormat());
}
private void scheduledTask() {
try {
Thread.sleep(5000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
然后項目啟動類上增加注解@EnableScheduling,表示開啟定時任務
package cn.wbnull.springbootdemo;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.scheduling.annotation.EnableScheduling;
@SpringBootApplication
@EnableScheduling
public class SpringBootDemoApplication {
public static void main(String[] args) {
SpringApplication.run(SpringBootDemoApplication.class, args);
}
}這里因為我們在ScheduledTask類創(chuàng)建了三個定時任務,@Scheduled默認是不并發(fā)執(zhí)行的,因此我們先注釋掉其他,分別進行測試。
1.1 @Scheduled(cron = “0/10 * * * * ?”)
package cn.wbnull.springbootdemo.schedule;
import cn.wbnull.springbootdemo.util.DateUtils;
import cn.wbnull.springbootdemo.util.LoggerUtils;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;
@Component
public class ScheduledTask {
@Scheduled(cron = "0/10 * * * * ?") //每10秒執(zhí)行一次
public void scheduledTaskByCorn() {
LoggerUtils.info("定時任務開始 ByCorn:" + DateUtils.dateFormat());
scheduledTask();
LoggerUtils.info("定時任務結束 ByCorn:" + DateUtils.dateFormat());
}
private void scheduledTask() {
try {
Thread.sleep(5000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}啟動項目,運行結果如下
[INFO][2019-02-18 16:08:40,095]||定時任務開始 ByCorn:2019-02-18 16:08:40
[INFO][2019-02-18 16:08:45,097]||定時任務結束 ByCorn:2019-02-18 16:08:45
[INFO][2019-02-18 16:08:50,001]||定時任務開始 ByCorn:2019-02-18 16:08:50
[INFO][2019-02-18 16:08:55,003]||定時任務結束 ByCorn:2019-02-18 16:08:55
[INFO][2019-02-18 16:09:00,002]||定時任務開始 ByCorn:2019-02-18 16:09:00
[INFO][2019-02-18 16:09:05,004]||定時任務結束 ByCorn:2019-02-18 16:09:05
[INFO][2019-02-18 16:09:10,001]||定時任務開始 ByCorn:2019-02-18 16:09:10
[INFO][2019-02-18 16:09:15,003]||定時任務結束 ByCorn:2019-02-18 16:09:15
[INFO][2019-02-18 16:09:20,001]||定時任務開始 ByCorn:2019-02-18 16:09:20
[INFO][2019-02-18 16:09:25,002]||定時任務結束 ByCorn:2019-02-18 16:09:25
[INFO][2019-02-18 16:09:30,001]||定時任務開始 ByCorn:2019-02-18 16:09:30
[INFO][2019-02-18 16:09:35,002]||定時任務結束 ByCorn:2019-02-18 16:09:35
我們再改下scheduledTask方法中線程休眠時間,使休眠時間大于定時任務間隔時間Thread.sleep(12000);,然后啟動項目,查看運行結果。
[INFO][2019-02-18 16:14:20,080]||定時任務開始 ByCorn:2019-02-18 16:14:20
[INFO][2019-02-18 16:14:32,081]||定時任務結束 ByCorn:2019-02-18 16:14:32
[INFO][2019-02-18 16:14:40,001]||定時任務開始 ByCorn:2019-02-18 16:14:40
[INFO][2019-02-18 16:14:52,002]||定時任務結束 ByCorn:2019-02-18 16:14:52
[INFO][2019-02-18 16:15:00,000]||定時任務開始 ByCorn:2019-02-18 16:15:00
[INFO][2019-02-18 16:15:12,002]||定時任務結束 ByCorn:2019-02-18 16:15:12
我們可以看到,對于cron表達式 來說,如果業(yè)務代碼執(zhí)行時間小于定時任務間隔時間,那么定時任務每10秒執(zhí)行一次,且不受業(yè)務代碼影響,無論業(yè)務代碼執(zhí)行多久,定時任務都是10秒執(zhí)行一次;
如果業(yè)務代碼執(zhí)行時間大于定時任務間隔時間,因定時任務默認不并發(fā),所以一直到業(yè)務代碼執(zhí)行完成的那個10秒,定時任務也是整10秒執(zhí)行一次,不受業(yè)務代碼影響。
注意:@Scheduled(cron = “0/10 * * * * ?”)控制的每10秒執(zhí)行一次的定時任務,是每10秒整執(zhí)行一次,即一分鐘內,如果當前秒數(shù)能夠整除10,則執(zhí)行定時任務,或理解為每分鐘0秒開始執(zhí)行,10秒后執(zhí)行下一次,執(zhí)行完一分鐘后,再從0秒開始。即只會在10s,20s,30s…的時候執(zhí)行,如果配置定時任務@Scheduled(cron = “0/7 * * * * ?”)這種,則只會在0s,7s,14s…的時候執(zhí)行。
1.2 @Scheduled(fixedRate = 10000)
package cn.wbnull.springbootdemo.schedule;
import cn.wbnull.springbootdemo.util.DateUtils;
import cn.wbnull.springbootdemo.util.LoggerUtils;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;
@Component
public class ScheduledTask {
@Scheduled(fixedRate = 10000) //每10秒執(zhí)行一次
public void scheduledTaskByFixedRate() {
LoggerUtils.info("定時任務開始 ByFixedRate:" + DateUtils.dateFormat());
scheduledTask();
LoggerUtils.info("定時任務結束 ByFixedRate:" + DateUtils.dateFormat());
}
private void scheduledTask() {
try {
Thread.sleep(12000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}啟動項目,運行結果如下
[INFO][2019-02-18 17:33:18,235]||定時任務開始 ByFixedRate:2019-02-18 17:33:18
[INFO][2019-02-18 17:33:23,239]||定時任務結束 ByFixedRate:2019-02-18 17:33:23
[INFO][2019-02-18 17:33:28,191]||定時任務開始 ByFixedRate:2019-02-18 17:33:28
[INFO][2019-02-18 17:33:33,195]||定時任務結束 ByFixedRate:2019-02-18 17:33:33
[INFO][2019-02-18 17:33:38,189]||定時任務開始 ByFixedRate:2019-02-18 17:33:38
[INFO][2019-02-18 17:33:43,191]||定時任務結束 ByFixedRate:2019-02-18 17:33:43
[INFO][2019-02-18 17:33:48,184]||定時任務開始 ByFixedRate:2019-02-18 17:33:48
[INFO][2019-02-18 17:33:53,186]||定時任務結束 ByFixedRate:2019-02-18 17:33:53
[INFO][2019-02-18 17:33:58,190]||定時任務開始 ByFixedRate:2019-02-18 17:33:58
[INFO][2019-02-18 17:34:03,193]||定時任務結束 ByFixedRate:2019-02-18 17:34:03
我們再改下scheduledTask方法中線程休眠時間,使休眠時間大于定時任務間隔時間Thread.sleep(12000);,然后啟動項目,查看運行結果。
[INFO][2019-02-18 17:31:30,122]||定時任務開始 ByFixedRate:2019-02-18 17:31:30
[INFO][2019-02-18 17:31:42,122]||定時任務結束 ByFixedRate:2019-02-18 17:31:42
[INFO][2019-02-18 17:31:42,123]||定時任務開始 ByFixedRate:2019-02-18 17:31:42
[INFO][2019-02-18 17:31:54,123]||定時任務結束 ByFixedRate:2019-02-18 17:31:54
[INFO][2019-02-18 17:31:54,124]||定時任務開始 ByFixedRate:2019-02-18 17:31:54
[INFO][2019-02-18 17:32:06,127]||定時任務結束 ByFixedRate:2019-02-18 17:32:06
[INFO][2019-02-18 17:32:06,127]||定時任務開始 ByFixedRate:2019-02-18 17:32:06
[INFO][2019-02-18 17:32:18,134]||定時任務結束 ByFixedRate:2019-02-18 17:32:18
對于fixedRate 來說,如果業(yè)務代碼執(zhí)行時間小于定時任務間隔時間,那么定時任務每10秒執(zhí)行一次,且不受業(yè)務代碼影響,無論業(yè)務代碼執(zhí)行多久,定時任務都是10秒執(zhí)行一次;
如果業(yè)務代碼執(zhí)行時間大于定時任務間隔時間,則定時任務循環(huán)執(zhí)行。
1.3 @Scheduled(fixedDelay = 10000)
package cn.wbnull.springbootdemo.schedule;
import cn.wbnull.springbootdemo.util.DateUtils;
import cn.wbnull.springbootdemo.util.LoggerUtils;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;
@Component
public class ScheduledTask {
@Scheduled(fixedDelay = 10000) //每10秒執(zhí)行一次
public void scheduledTaskByFixedDelay() {
LoggerUtils.info("定時任務開始 ByFixedDelay:" + DateUtils.dateFormat());
scheduledTask();
LoggerUtils.info("定時任務結束 ByFixedDelay:" + DateUtils.dateFormat());
}
private void scheduledTask() {
try {
Thread.sleep(5000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}啟動項目,運行結果如下
[INFO][2019-02-18 17:45:30,784]||定時任務開始 ByFixedDelay:2019-02-18 17:45:30
[INFO][2019-02-18 17:45:35,792]||定時任務結束 ByFixedDelay:2019-02-18 17:45:35
[INFO][2019-02-18 17:45:45,803]||定時任務開始 ByFixedDelay:2019-02-18 17:45:45
[INFO][2019-02-18 17:45:50,812]||定時任務結束 ByFixedDelay:2019-02-18 17:45:50
[INFO][2019-02-18 17:46:00,814]||定時任務開始 ByFixedDelay:2019-02-18 17:46:00
[INFO][2019-02-18 17:46:05,817]||定時任務結束 ByFixedDelay:2019-02-18 17:46:05
[INFO][2019-02-18 17:46:15,821]||定時任務開始 ByFixedDelay:2019-02-18 17:46:15
[INFO][2019-02-18 17:46:20,825]||定時任務結束 ByFixedDelay:2019-02-18 17:46:20
[INFO][2019-02-18 17:46:30,829]||定時任務開始 ByFixedDelay:2019-02-18 17:46:30
[INFO][2019-02-18 17:46:35,834]||定時任務結束 ByFixedDelay:2019-02-18 17:46:35
我們再改下scheduledTask方法中線程休眠時間,使休眠時間大于定時任務間隔時間Thread.sleep(12000);,然后啟動項目,查看運行結果。
[INFO][2019-02-18 17:47:06,871]||定時任務開始 ByFixedDelay:2019-02-18 17:47:06
[INFO][2019-02-18 17:47:18,879]||定時任務結束 ByFixedDelay:2019-02-18 17:47:18
[INFO][2019-02-18 17:47:28,890]||定時任務開始 ByFixedDelay:2019-02-18 17:47:28
[INFO][2019-02-18 17:47:40,896]||定時任務結束 ByFixedDelay:2019-02-18 17:47:40
[INFO][2019-02-18 17:47:50,903]||定時任務開始 ByFixedDelay:2019-02-18 17:47:50
[INFO][2019-02-18 17:48:02,911]||定時任務結束 ByFixedDelay:2019-02-18 17:48:02
[INFO][2019-02-18 17:48:12,917]||定時任務開始 ByFixedDelay:2019-02-18 17:48:12
[INFO][2019-02-18 17:48:24,924]||定時任務結束 ByFixedDelay:2019-02-18 17:48:24
對于fixedDelay 來說,不管業(yè)務代碼執(zhí)行時間與定時任務間隔時間熟長熟短,定時任務都會等業(yè)務代碼執(zhí)行完成后再開啟新一輪定時。
不過,一般大家在使用定時任務時,都是定時任務時間間隔大于業(yè)務代碼執(zhí)行時間。
1.4 多說一點
對于固定時間執(zhí)行的定時任務,比如每天凌晨4點執(zhí)行,只能使用cron表達式的方式
2. corn表達式
2.1 corn表達式格式
corn表達式格式:秒 分 時 日 月 星期 年(可選)
| 字段名 | 允許的值 | 允許的特殊字符 |
|---|---|---|
| 秒 | 0-59 | , - * / |
| 分 | 0-59 | , - * / |
| 時 | 0-23 | , - * / |
| 日 | 1-31 | , - * ? / L W C |
| 月 | 1-12 或 JAN-DEC | , - * / |
| 星期 | 1-7 或 SUN-SAT | , - * ? / L C # |
| 年(可選) | 空 或 1970-2099 | , - * / |
釋義:
1、*:通配符,表示該字段可以接收任意值。
2、? :表示不確定的值,或不關心它為何值,僅在日期和星期中使用,當其中一個設置了條件時,另外一個用"?" 來表示"任何值"。
3、,:表示多個值,附加一個生效的值。
4、-:表示一個指定的范圍
5、/:指定一個值的增量值。例n/m表示從n開始,每次增加m
6、L:用在日期表示當月的最后一天,用在星期"L"單獨使用時就等于"7"或"SAT",如果和數(shù)字聯(lián)合使用表示該月最后一個星期X。例如,"0L"表示該月最后一個星期日。
7、W:指定離給定日期最近的工作日(周一到周五),可以用"LW"表示該月最后一個工作日。例如,"10W"表示這個月離10號最近的工作日
8、C:表示和calendar聯(lián)系后計算過的值。例如:用在日期中,"5C"表示該月第5天或之后包括calendar的第一天;用在星期中,"5C"表示這周四或之后包括calendar的第 一天。
9、#:表示該月第幾個星期X。例6#3表示該月第三個周五。
2.2 示例值
0 * * * * ? 每分鐘觸發(fā)
0 0 * * * ? 每小時整觸發(fā)
0 0 4 * * ? 每天凌晨4點觸發(fā)
0 15 10 * * ? 每天早上10:15觸發(fā)
*/5 * * * * ? 每隔5秒觸發(fā)
0 */5 * * * ? 每隔5分鐘觸發(fā)
0 0 4 1 * ? 每月1號凌晨4點觸發(fā)
0 0 4 L * ? 每月最后一天凌晨3點觸發(fā)
0 0 3 ? * L 每周星期六凌晨3點觸發(fā)
0 11,22,33 * * * ? 每小時11分、22分、33分觸發(fā)
3. 配置定時任務
對于上面那些簡單的定時任務,定時任務的corn表達式寫死在代碼里,如果要改動表達式,需要修改代碼,重新打包發(fā)布,比較麻煩。因此,我們可以把corn表達式配置在配置文件中,然后程序讀取配置,當需要修改表達式時,只需要修改配置文件即可。
application.yml增加配置
demo: corn: 0/11 * * * * ?
定時任務
package cn.wbnull.springbootdemo.schedule;
import cn.wbnull.springbootdemo.util.DateUtils;
import cn.wbnull.springbootdemo.util.LoggerUtils;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;
@Component
public class ScheduledTask {
@Scheduled(cron = "${demo.corn}")
public void scheduledTaskByConfig() {
LoggerUtils.info("定時任務 ByConfig:" + DateUtils.dateFormat());
}
}啟動項目,運行結果如下
[INFO][2019-02-18 23:47:33,047]||定時任務 ByConfig:2019-02-18 23:47:33
[INFO][2019-02-18 23:47:44,003]||定時任務 ByConfig:2019-02-18 23:47:44
[INFO][2019-02-18 23:47:55,009]||定時任務 ByConfig:2019-02-18 23:47:55
[INFO][2019-02-18 23:48:00,008]||定時任務 ByConfig:2019-02-18 23:48:00
[INFO][2019-02-18 23:48:11,009]||定時任務 ByConfig:2019-02-18 23:48:11
[INFO][2019-02-18 23:48:22,009]||定時任務 ByConfig:2019-02-18 23:48:22
[INFO][2019-02-18 23:48:33,009]||定時任務 ByConfig:2019-02-18 23:48:33
修改application.yml配置
demo: corn: 0/23 * * * * ?
啟動項目,運行結果如下
[INFO][2019-02-18 23:52:23,089]||定時任務 ByConfig:2019-02-18 23:52:23
[INFO][2019-02-18 23:52:46,008]||定時任務 ByConfig:2019-02-18 23:52:46
[INFO][2019-02-18 23:53:00,009]||定時任務 ByConfig:2019-02-18 23:53:00
[INFO][2019-02-18 23:53:23,002]||定時任務 ByConfig:2019-02-18 23:53:23
[INFO][2019-02-18 23:53:46,009]||定時任務 ByConfig:2019-02-18 23:53:46
定時任務根據(jù)配置文件動態(tài)變化。
4. 動態(tài)修改定時任務
對于有些情況,我們需要在代碼中,通過方法動態(tài)修改定時任務corn表達式
application.yml配置
demo: corn: 0/7 * * * * ? cornV2: 0/22 * * * * ?
新建ScheduledTaskV2.java
package cn.wbnull.springbootdemo.schedule;
import cn.wbnull.springbootdemo.util.DateUtils;
import cn.wbnull.springbootdemo.util.LoggerUtils;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.scheduling.annotation.SchedulingConfigurer;
import org.springframework.scheduling.config.ScheduledTaskRegistrar;
import org.springframework.scheduling.support.CronTrigger;
import org.springframework.stereotype.Component;
@Component
public class ScheduledTaskV2 implements SchedulingConfigurer {
@Value("${demo.corn}")
private String corn;
@Value("${demo.cornV2}")
private String cornV2;
private int tag = 0;
@Override
public void configureTasks(ScheduledTaskRegistrar scheduledTaskRegistrar) {
scheduledTaskRegistrar.addTriggerTask(() -> {
LoggerUtils.info("定時任務V2:" + DateUtils.dateFormat());
}, (triggerContext) -> {
CronTrigger cronTrigger;
if (tag % 2 == 0) {
LoggerUtils.info("定時任務V2動態(tài)修改corn表達式:" + corn + "," + DateUtils.dateFormat());
cronTrigger = new CronTrigger(corn);
tag++;
} else {
LoggerUtils.info("定時任務V2動態(tài)修改corn表達式:" + cornV2 + "," + DateUtils.dateFormat());
cronTrigger = new CronTrigger(cornV2);
tag++;
}
return cronTrigger.nextExecutionTime(triggerContext);
});
}
}
啟動項目,運行結果如下
[INFO][2019-02-19 00:19:49,011]||定時任務V2:2019-02-19 00:19:49
[INFO][2019-02-19 00:19:49,011]||定時任務V2動態(tài)修改corn表達式:0/22 * * * * ?,2019-02-19 00:19:49
[INFO][2019-02-19 00:20:00,007]||定時任務V2:2019-02-19 00:20:00
[INFO][2019-02-19 00:20:00,007]||定時任務V2動態(tài)修改corn表達式:0/7 * * * * ?,2019-02-19 00:20:00
[INFO][2019-02-19 00:20:07,006]||定時任務V2:2019-02-19 00:20:07
[INFO][2019-02-19 00:20:07,006]||定時任務V2動態(tài)修改corn表達式:0/22 * * * * ?,2019-02-19 00:20:07
[INFO][2019-02-19 00:20:22,008]||定時任務V2:2019-02-19 00:20:22
[INFO][2019-02-19 00:20:22,008]||定時任務V2動態(tài)修改corn表達式:0/7 * * * * ?,2019-02-19 00:20:22
[INFO][2019-02-19 00:20:28,010]||定時任務V2:2019-02-19 00:20:28
[INFO][2019-02-19 00:20:28,010]||定時任務V2動態(tài)修改corn表達式:0/22 * * * * ?,2019-02-19 00:20:28
[INFO][2019-02-19 00:20:44,003]||定時任務V2:2019-02-19 00:20:44
[INFO][2019-02-19 00:20:44,003]||定時任務V2動態(tài)修改corn表達式:0/7 * * * * ?,2019-02-19 00:20:44
[INFO][2019-02-19 00:20:49,004]||定時任務V2:2019-02-19 00:20:49
[INFO][2019-02-19 00:20:49,004]||定時任務V2動態(tài)修改corn表達式:0/22 * * * * ?,2019-02-19 00:20:49
[INFO][2019-02-19 00:21:00,011]||定時任務V2:2019-02-19 00:21:00
[INFO][2019-02-19 00:21:00,011]||定時任務V2動態(tài)修改corn表達式:0/7 * * * * ?,2019-02-19 00:21:00
[INFO][2019-02-19 00:21:07,011]||定時任務V2:2019-02-19 00:21:07
[INFO][2019-02-19 00:21:07,011]||定時任務V2動態(tài)修改corn表達式:0/22 * * * * ?,2019-02-19 00:21:07
成功通過代碼動態(tài)修改corn表達式且運行結果正確。
5. 并發(fā)執(zhí)行定時任務
回到我們 1. 簡單定時任務 中創(chuàng)建的三個定時任務,當時因為@Scheduled默認是不并發(fā)執(zhí)行的,所以我們先注釋掉了其他定時任務,分別進行的測試。
那我們實際開發(fā)中,確實創(chuàng)建了多個定時任務,且想并發(fā)執(zhí)行時,該怎么做呢?
定時任務類添加注解@EnableAsync,需并發(fā)執(zhí)行的定時任務方法添加注解@Async
新建定時任務類ScheduledTaskV3
package cn.wbnull.springbootdemo.schedule;
import cn.wbnull.springbootdemo.util.DateUtils;
import cn.wbnull.springbootdemo.util.LoggerUtils;
import org.springframework.scheduling.annotation.Async;
import org.springframework.scheduling.annotation.EnableAsync;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;
@Component
@EnableAsync
public class ScheduledTaskV3 {
@Scheduled(cron = "0/7 * * * * ?")
@Async
public void scheduledTaskV1() {
LoggerUtils.info("定時任務V3,定時任務1開始:" + DateUtils.dateFormat());
scheduledTask();
LoggerUtils.info("定時任務V3,定時任務1結束:" + DateUtils.dateFormat());
}
@Scheduled(cron = "0/10 * * * * ?")
@Async
public void scheduledTaskV2() {
LoggerUtils.info("定時任務V3,定時任務2開始:" + DateUtils.dateFormat());
scheduledTask();
LoggerUtils.info("定時任務V3,定時任務2結束:" + DateUtils.dateFormat());
}
@Scheduled(cron = "0/22 * * * * ?")
@Async
public void scheduledTaskV3() {
LoggerUtils.info("定時任務V3,定時任務3開始:" + DateUtils.dateFormat());
scheduledTask();
LoggerUtils.info("定時任務V3,定時任務3結束:" + DateUtils.dateFormat());
}
private void scheduledTask() {
try {
Thread.sleep(5000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
啟動項目,運行結果如下
[INFO][2019-02-19 00:36:21,077]||定時任務V3,定時任務1開始:2019-02-19 00:36:21
[INFO][2019-02-19 00:36:22,003]||定時任務V3,定時任務3開始:2019-02-19 00:36:22
[INFO][2019-02-19 00:36:26,078]||定時任務V3,定時任務1結束:2019-02-19 00:36:26
[INFO][2019-02-19 00:36:27,006]||定時任務V3,定時任務3結束:2019-02-19 00:36:27
[INFO][2019-02-19 00:36:28,003]||定時任務V3,定時任務1開始:2019-02-19 00:36:28
[INFO][2019-02-19 00:36:30,003]||定時任務V3,定時任務2開始:2019-02-19 00:36:30
[INFO][2019-02-19 00:36:33,003]||定時任務V3,定時任務1結束:2019-02-19 00:36:33
[INFO][2019-02-19 00:36:35,004]||定時任務V3,定時任務1開始:2019-02-19 00:36:35
[INFO][2019-02-19 00:36:35,005]||定時任務V3,定時任務2結束:2019-02-19 00:36:35
[INFO][2019-02-19 00:36:40,003]||定時任務V3,定時任務2開始:2019-02-19 00:36:40
[INFO][2019-02-19 00:36:40,005]||定時任務V3,定時任務1結束:2019-02-19 00:36:40
[INFO][2019-02-19 00:36:42,001]||定時任務V3,定時任務1開始:2019-02-19 00:36:42
[INFO][2019-02-19 00:36:44,003]||定時任務V3,定時任務3開始:2019-02-19 00:36:44
[INFO][2019-02-19 00:36:45,004]||定時任務V3,定時任務2結束:2019-02-19 00:36:45
[INFO][2019-02-19 00:36:47,002]||定時任務V3,定時任務1結束:2019-02-19 00:36:47
[INFO][2019-02-19 00:36:49,002]||定時任務V3,定時任務1開始:2019-02-19 00:36:49
[INFO][2019-02-19 00:36:49,004]||定時任務V3,定時任務3結束:2019-02-19 00:36:49
[INFO][2019-02-19 00:36:50,001]||定時任務V3,定時任務2開始:2019-02-19 00:36:50
定時任務能夠并發(fā)執(zhí)行。
GitHub:https://github.com/dkbnull/SpringBootDemo
微信:https://mp.weixin.qq.com/s/qJZpCjcZYQWnmDkztiA_uQ
微博:https://weibo.com/ttarticle/p/show?id=2309404446645717696521
知乎:https://zhuanlan.zhihu.com/p/95813468
到此這篇關于Spring Boot使用Schedule實現(xiàn)定時任務的文章就介紹到這了,更多相關Spring Boot Schedule定時任務內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
MyBatis測試報錯:Cannot?determine?value?type?from?string?&a
這篇文章主要給大家介紹了關于MyBatis測試報錯:Cannot?determine?value?type?from?string?'xxx'的解決辦法,文中通過實例代碼介紹的非常詳細,需要的朋友可以參考下2023-02-02
java動態(tài)構建數(shù)據(jù)庫復雜查詢教程
這篇文章主要介紹了java動態(tài)構建數(shù)據(jù)庫復雜查詢的實現(xiàn)示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2021-11-11

