欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

SpringBoot集成Quartz實(shí)現(xiàn)定時(shí)任務(wù)的方法

 更新時(shí)間:2020年05月06日 09:31:44   作者:民意代表  
Quartz是一個(gè)定時(shí)任務(wù)框架,其他介紹網(wǎng)上也很詳盡。這里要介紹一下Quartz里的幾個(gè)非常核心的接口。通過實(shí)例代碼給大家講解SpringBoot集成Quartz實(shí)現(xiàn)定時(shí)任務(wù)的方法,感興趣的朋友一起看看吧

1 需求

在我的前后端分離的實(shí)驗(yàn)室管理項(xiàng)目中,有一個(gè)功能是學(xué)生狀態(tài)統(tǒng)計(jì)。我的設(shè)計(jì)是按天統(tǒng)計(jì)每種狀態(tài)的比例。為了便于計(jì)算,在每天0點(diǎn),系統(tǒng)需要將學(xué)生的狀態(tài)重置,并插入一條數(shù)據(jù)作為一天的開始狀態(tài)。另外,考慮到學(xué)生的請(qǐng)假需求,請(qǐng)假的申請(qǐng)往往是提前做好,等系統(tǒng)時(shí)間走到實(shí)際請(qǐng)假時(shí)間的時(shí)候,系統(tǒng)要將學(xué)生的狀態(tài)修改為請(qǐng)假。

顯然,這兩個(gè)子需求都可以通過定時(shí)任務(wù)實(shí)現(xiàn)。在網(wǎng)上略做搜索以后,我選擇了比較流行的定時(shí)任務(wù)框架Quartz。

2 Quartz

Quartz是一個(gè)定時(shí)任務(wù)框架,其他介紹網(wǎng)上也很詳盡。這里要介紹一下Quartz里的幾個(gè)非常核心的接口。

2.1 Scheduler接口

Scheduler翻譯成調(diào)度器,Quartz通過調(diào)度器來注冊(cè)、暫停、刪除Trigger和JobDetail。Scheduler還擁有一個(gè)SchedulerContext,顧名思義就是上下文,通過SchedulerContext我們可以獲取到觸發(fā)器和任務(wù)的一些信息。

2.2 Trigger接口

Trigger可以翻譯成觸發(fā)器,通過cron表達(dá)式或是SimpleScheduleBuilder等類,指定任務(wù)執(zhí)行的周期。系統(tǒng)時(shí)間走到觸發(fā)器指定的時(shí)間的時(shí)候,觸發(fā)器就會(huì)觸發(fā)任務(wù)的執(zhí)行。

2.3 JobDetail接口

Job接口是真正需要執(zhí)行的任務(wù)。JobDetail接口相當(dāng)于將Job接口包裝了一下,Trigger和Scheduler實(shí)際用到的都是JobDetail。

3 SpringBoot官方文檔解讀

SpringBoot官方寫了spring-boot-starter-quartz。使用過SpringBoot的同學(xué)都知道這是一個(gè)官方提供的啟動(dòng)器,有了這個(gè)啟動(dòng)器,集成的操作就會(huì)被大大簡(jiǎn)化。

現(xiàn)在我們來看一看SpingBoot2.2.6官方文檔,其中第4.20小節(jié)Quartz Scheduler就談到了Quartz,但很可惜一共只有兩頁不到的內(nèi)容,先來看看這么精華的文檔里能學(xué)到些什么。

Spring Boot offers several conveniences for working with the Quartz scheduler, including the
spring-boot-starter-quartz “Starter”. If Quartz is available, a Scheduler is auto-configured (through the SchedulerFactoryBean abstraction).
Beans of the following types are automatically picked up and associated with the Scheduler:
• JobDetail: defines a particular Job. JobDetail instances can be built with the JobBuilder API.
• Calendar.
• Trigger: defines when a particular job is triggered.

翻譯一下:

SpringBoot提供了一些便捷的方法來和Quartz協(xié)同工作,這些方法里面包括`spring-boot-starter-quartz`這個(gè)啟動(dòng)器。如果Quartz可用,Scheduler會(huì)通過SchedulerFactoryBean這個(gè)工廠bean自動(dòng)配置到SpringBoot里。
JobDetail、Calendar、Trigger這些類型的bean會(huì)被自動(dòng)采集并關(guān)聯(lián)到Scheduler上。

Jobs can define setters to inject data map properties. Regular beans can also be injected in a similar manner.

翻譯一下:

Job可以定義setter(也就是set方法)來注入配置信息。也可以用同樣的方法注入普通的bean。

下面是文檔里給的示例代碼,我直接完全照著寫,拿到的卻是null。不知道是不是我的使用方式有誤。后來仔細(xì)一想,文檔的意思應(yīng)該是在創(chuàng)建Job對(duì)象之后,調(diào)用set方法將依賴注入進(jìn)去。但后面我們是通過框架反射生成的Job對(duì)象,這樣做反而會(huì)搞得更加復(fù)雜。最后還是決定采用給Job類加@Component注解的方法。

文檔的其他篇幅就介紹了一些配置,但是介紹得也不全面,看了幫助也并不是很大。詳細(xì)的配置可以參考w3school的Quartz配置。

4 SpringBoot集成Quartz

4.1 建表

我選擇將定時(shí)任務(wù)的信息保存在數(shù)據(jù)庫(kù)中,優(yōu)點(diǎn)是顯而易見的,定時(shí)任務(wù)不會(huì)因?yàn)橄到y(tǒng)的崩潰而丟失。

建表的sql語句在Quartz的github中可以找到,里面有針對(duì)每一種常用數(shù)據(jù)庫(kù)的sql語句,具體地址是:Quartz數(shù)據(jù)庫(kù)建表sql。

建表以后,可以看到數(shù)據(jù)庫(kù)里多了11張表。我們完全不需要關(guān)心每張表的具體作用,在添加刪除任務(wù)、觸發(fā)器等的時(shí)候,Quartz框架會(huì)操作這些表。

4.2 引入依賴

pom.xml里添加依賴。

<!-- quartz 定時(shí)任務(wù) -->
<dependency>
 <groupId>org.springframework.boot</groupId>
 <artifactId>spring-boot-starter-quartz</artifactId>
 <version>2.2.6.RELEASE</version>
</dependency>

4.3 配置quartz

application.yml中配置quartz。相關(guān)配置的作用已經(jīng)寫在注解上。

# spring的datasource等配置未貼出
spring:
 quartz:
  # 將任務(wù)等保存化到數(shù)據(jù)庫(kù)
  job-store-type: jdbc
  # 程序結(jié)束時(shí)會(huì)等待quartz相關(guān)的內(nèi)容結(jié)束
  wait-for-jobs-to-complete-on-shutdown: true
  # QuartzScheduler啟動(dòng)時(shí)更新己存在的Job,這樣就不用每次修改targetObject后刪除qrtz_job_details表對(duì)應(yīng)記錄
  overwrite-existing-jobs: true
  # 這里居然是個(gè)map,搞得智能提示都沒有,佛了
  properties:
  org:
   quartz:
   	# scheduler相關(guān)
   scheduler:
    # scheduler的實(shí)例名
    instanceName: scheduler
    instanceId: AUTO
   # 持久化相關(guān)
   jobStore:
    class: org.quartz.impl.jdbcjobstore.JobStoreTX
    driverDelegateClass: org.quartz.impl.jdbcjobstore.StdJDBCDelegate
    # 表示數(shù)據(jù)庫(kù)中相關(guān)表是QRTZ_開頭的
    tablePrefix: QRTZ_
    useProperties: false
   # 線程池相關(guān)
   threadPool:
    class: org.quartz.simpl.SimpleThreadPool
    # 線程數(shù)
    threadCount: 10
    # 線程優(yōu)先級(jí)
    threadPriority: 5
    threadsInheritContextClassLoaderOfInitializingThread: true

4.4 注冊(cè)周期性的定時(shí)任務(wù)

第1節(jié)中提到的第一個(gè)子需求是在每天0點(diǎn)執(zhí)行的,是一個(gè)周期性的任務(wù),任務(wù)內(nèi)容也是確定的,所以直接在代碼里注冊(cè)JobDetail和Trigger的bean就可以了。當(dāng)然,這些JobDetail和Trigger也是會(huì)被持久化到數(shù)據(jù)庫(kù)里。

/**
 * Quartz的相關(guān)配置,注冊(cè)JobDetail和Trigger
 * 注意JobDetail和Trigger是org.quartz包下的,不是spring包下的,不要導(dǎo)入錯(cuò)誤
 */
@Configuration
public class QuartzConfig {

 @Bean
 public JobDetail jobDetail() {
  JobDetail jobDetail = JobBuilder.newJob(StartOfDayJob.class)
    .withIdentity("start_of_day", "start_of_day")
    .storeDurably()
    .build();
  return jobDetail;
 }

 @Bean
 public Trigger trigger() {
  Trigger trigger = TriggerBuilder.newTrigger()
    .forJob(jobDetail())
    .withIdentity("start_of_day", "start_of_day")
    .startNow()
    // 每天0點(diǎn)執(zhí)行
    .withSchedule(CronScheduleBuilder.cronSchedule("0 0 0 * * ?"))
    .build();
  return trigger;
 }
}

builder類創(chuàng)建了一個(gè)JobDetail和一個(gè)Trigger并注冊(cè)成為Spring bean。從第3節(jié)中摘錄的官方文檔中,我們已經(jīng)知道這些bean會(huì)自動(dòng)關(guān)聯(lián)到調(diào)度器上。需要注意的是JobDetail和Trigger需要設(shè)置組名和自己的名字,用來作為唯一標(biāo)識(shí)。當(dāng)然,JobDetail和Trigger的唯一標(biāo)識(shí)可以相同,因?yàn)樗麄兪遣煌念悺?/p>

Trigger通過cron表達(dá)式指定了任務(wù)執(zhí)行的周期。對(duì)cron表達(dá)式不熟悉的同學(xué)可以百度學(xué)習(xí)一下。

JobDetail里有一個(gè)StartOfDayJob類,這個(gè)類就是Job接口的一個(gè)實(shí)現(xiàn)類,里面定義了任務(wù)的具體內(nèi)容,看一下代碼:

@Component
public class StartOfDayJob extends QuartzJobBean {
 private StudentService studentService;

 @Autowired
 public StartOfDayJob(StudentService studentService) {
  this.studentService = studentService;
 }

 @Override
 protected void executeInternal(JobExecutionContext jobExecutionContext)
   throws JobExecutionException {
  // 任務(wù)的具體邏輯
 }
}

這里面有一個(gè)小問題,上面用builder創(chuàng)建JobDetail時(shí),傳入了StartOfDayJob.class,按常理推測(cè),應(yīng)該是Quartz框架通過反射創(chuàng)建StartOfDayJob對(duì)象,再調(diào)用executeInternal()執(zhí)行任務(wù)。這樣依賴,這個(gè)Job是Quartz通過反射創(chuàng)建的,即使加了注解@Component,這個(gè)StartOfDayJob對(duì)象也不會(huì)被注冊(cè)到ioc容器中,更不可能實(shí)現(xiàn)依賴的自動(dòng)裝配。

網(wǎng)上很多博客也是這么介紹的。但是根據(jù)我的實(shí)際測(cè)試,這樣寫可以完成依賴注入,但我還不知道它的實(shí)現(xiàn)原理。

4.5 注冊(cè)無周期性的定時(shí)任務(wù)

第1節(jié)中提到的第二個(gè)子需求是學(xué)生請(qǐng)假,顯然請(qǐng)假是不定時(shí)的,一次性的,而且不具有周期性。

4.5節(jié)與4.4節(jié)大體相同,但是有兩點(diǎn)區(qū)別:

  • Job類需要獲取到一些數(shù)據(jù)用于任務(wù)的執(zhí)行;任務(wù)執(zhí)行完成后刪除Job和Trigger。
  • 業(yè)務(wù)邏輯是在老師批準(zhǔn)學(xué)生的請(qǐng)假申請(qǐng)時(shí),向調(diào)度器添加Trigger和JobDetail。

實(shí)體類:

public class LeaveApplication {
 @TableId(type = IdType.AUTO)
 private Integer id;
 private Long proposerUsername;
 @JsonFormat( pattern = "yyyy-MM-dd HH:mm",timezone="GMT+8")
 private LocalDateTime startTime;
 @JsonFormat( pattern = "yyyy-MM-dd HH:mm",timezone="GMT+8")
 private LocalDateTime endTime;
 private String reason;
 private String state;
 private String disapprovedReason;
 private Long checkerUsername;
 private LocalDateTime checkTime;

 // 省略getter、setter
}

Service層邏輯,重要的地方已在注釋中說明。

@Service
public class LeaveApplicationServiceImpl implements LeaveApplicationService {
 @Autowired
 private Scheduler scheduler;
 
 // 省略其他方法與其他依賴

 /**
  * 添加job和trigger到scheduler
  */
 private void addJobAndTrigger(LeaveApplication leaveApplication) {
  Long proposerUsername = leaveApplication.getProposerUsername();
  // 創(chuàng)建請(qǐng)假開始Job
  LocalDateTime startTime = leaveApplication.getStartTime();
  JobDetail startJobDetail = JobBuilder.newJob(LeaveStartJob.class)
   	// 指定任務(wù)組名和任務(wù)名
    .withIdentity(leaveApplication.getStartTime().toString(),
      proposerUsername + "_start")
    // 添加一些參數(shù),執(zhí)行的時(shí)候用
    .usingJobData("username", proposerUsername)
    .usingJobData("time", startTime.toString())
    .build();
  // 創(chuàng)建請(qǐng)假開始任務(wù)的觸發(fā)器
  // 創(chuàng)建cron表達(dá)式指定任務(wù)執(zhí)行的時(shí)間,由于請(qǐng)假時(shí)間是確定的,所以年月日時(shí)分秒都是確定的,這也符合任務(wù)只執(zhí)行一次的要求。
  String startCron = String.format("%d %d %d %d %d ? %d",
    startTime.getSecond(),
    startTime.getMinute(),
    startTime.getHour(),
    startTime.getDayOfMonth(),
    startTime.getMonth().getValue(),
    startTime.getYear());
  CronTrigger startCronTrigger = TriggerBuilder.newTrigger()
	   // 指定觸發(fā)器組名和觸發(fā)器名
    .withIdentity(leaveApplication.getStartTime().toString(),
      proposerUsername + "_start")
    .withSchedule(CronScheduleBuilder.cronSchedule(startCron))
    .build();

  // 將job和trigger添加到scheduler里
  try {
   scheduler.scheduleJob(startJobDetail, startCronTrigger);
  } catch (SchedulerException e) {
   e.printStackTrace();
   throw new CustomizedException("添加請(qǐng)假任務(wù)失敗");
  }
 }
}

Job類邏輯,重要的地方已在注釋中說明。

@Component
public class LeaveStartJob extends QuartzJobBean {
 private Scheduler scheduler;
 private SystemUserMapperPlus systemUserMapperPlus;

 @Autowired
 public LeaveStartJob(Scheduler scheduler,
       SystemUserMapperPlus systemUserMapperPlus) {
  this.scheduler = scheduler;
  this.systemUserMapperPlus = systemUserMapperPlus;
 }

 @Override
 protected void executeInternal(JobExecutionContext jobExecutionContext)
   throws JobExecutionException {
  Trigger trigger = jobExecutionContext.getTrigger();
  JobDetail jobDetail = jobExecutionContext.getJobDetail();
  JobDataMap jobDataMap = jobDetail.getJobDataMap();
  // 將添加任務(wù)的時(shí)候存進(jìn)去的數(shù)據(jù)拿出來
  long username = jobDataMap.getLongValue("username");
  LocalDateTime time = LocalDateTime.parse(jobDataMap.getString("time"));

  // 編寫任務(wù)的邏輯

  // 執(zhí)行之后刪除任務(wù)
  try {
   // 暫停觸發(fā)器的計(jì)時(shí)
   scheduler.pauseTrigger(trigger.getKey());
   // 移除觸發(fā)器中的任務(wù)
   scheduler.unscheduleJob(trigger.getKey());
   // 刪除任務(wù)
   scheduler.deleteJob(jobDetail.getKey());
  } catch (SchedulerException e) {
   e.printStackTrace();
  }
 }
}

5 總結(jié)

上文所述的內(nèi)容應(yīng)該可以滿足絕大部分定時(shí)任務(wù)的需求。我在查閱網(wǎng)上的博客之后,發(fā)現(xiàn)大部分博客里介紹的Quartz使用還是停留在Spring階段,配置也都是通過xml,因此我在實(shí)現(xiàn)了功能以后,將整個(gè)過程總結(jié)了一下,留給需要的人以及以后的自己做參考。

總體上來說,Quartz實(shí)現(xiàn)定時(shí)任務(wù)還是非常方便的,與SpringBoot整合之后配置也非常簡(jiǎn)單,是實(shí)現(xiàn)定時(shí)任務(wù)的不錯(cuò)的選擇。

5.2 小坑1

在IDEA2020.1版本里使用SpringBoot與Quartz時(shí),報(bào)錯(cuò)找不到org.quartz程序包,但是依賴?yán)锩婷髅饔衞rg.quartz,類里的import也沒有報(bào)錯(cuò),還可以通過Ctrl+鼠標(biāo)左鍵直接跳轉(zhuǎn)到相應(yīng)的類里。后面我用了IDEA2019.3.4就不再有這個(gè)錯(cuò)誤。那么就是新版IDEA的BUG了。

到此這篇關(guān)于SpringBoot集成Quartz實(shí)現(xiàn)定時(shí)任務(wù)的文章就介紹到這了,更多相關(guān)SpringBoot集成Quartz實(shí)現(xiàn)定時(shí)任務(wù)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 微信企業(yè)號(hào) 根據(jù)錯(cuò)誤碼返回錯(cuò)誤信息類封裝

    微信企業(yè)號(hào) 根據(jù)錯(cuò)誤碼返回錯(cuò)誤信息類封裝

    這篇文章主要介紹了微信企業(yè)號(hào) 根據(jù)錯(cuò)誤碼返回錯(cuò)誤信息類封裝的相關(guān)資料,需要的朋友可以參考下
    2016-10-10
  • NoHttpResponseException問題分析解決記錄

    NoHttpResponseException問題分析解決記錄

    這篇文章主要為大家介紹了NoHttpResponseException問題分析解決記錄,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-08-08
  • springMVC盜鏈接詳解

    springMVC盜鏈接詳解

    這篇文章主要為大家詳細(xì)介紹了SpringMVC盜鏈接詳解,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能給你帶來幫助
    2021-07-07
  • MyBatis實(shí)現(xiàn)遞歸查詢的方法詳解

    MyBatis實(shí)現(xiàn)遞歸查詢的方法詳解

    在項(xiàng)目開發(fā)過程中,往往會(huì)遇到多級(jí)菜單、分類等多層級(jí)結(jié)構(gòu)數(shù)據(jù)的查詢。本文就來為大家講講MyBatis實(shí)現(xiàn)遞歸查詢的方法,感興趣的可以動(dòng)手嘗試一下
    2022-08-08
  • Java實(shí)現(xiàn)下載文件的6種方式

    Java實(shí)現(xiàn)下載文件的6種方式

    本文主要介紹了Java實(shí)現(xiàn)下載文件的6種方式,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-06-06
  • springboot + devtools(熱部署)實(shí)例教程

    springboot + devtools(熱部署)實(shí)例教程

    devtools是boot的一個(gè)熱部署工具,當(dāng)我們修改了classpath下的文件(包括類文件、屬性文件、頁面等)時(shí),會(huì)重新啟動(dòng)應(yīng)用。本文通過實(shí)例給大家介紹springboot+devtools熱部署,感興趣的朋友一起看看吧
    2017-04-04
  • SpringMVC上傳文件的兩種方法

    SpringMVC上傳文件的兩種方法

    這篇文章主要為大家詳細(xì)介紹了SpringMVC上傳文件的兩種方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-11-11
  • Java反射(JDK)與動(dòng)態(tài)代理(CGLIB)詳解

    Java反射(JDK)與動(dòng)態(tài)代理(CGLIB)詳解

    下面小編就為大家?guī)硪黄獪\談Java反射與動(dòng)態(tài)代理。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2021-08-08
  • Java實(shí)現(xiàn)爬取百度圖片的方法分析

    Java實(shí)現(xiàn)爬取百度圖片的方法分析

    這篇文章主要介紹了Java實(shí)現(xiàn)爬取百度圖片的方法,結(jié)合實(shí)例形式分析了java基于jsonp爬取百度圖片的相關(guān)原理、操作技巧與注意事項(xiàng),需要的朋友可以參考下
    2018-12-12
  • Spring如何利用@Value注解讀取yml中的map配置

    Spring如何利用@Value注解讀取yml中的map配置

    這篇文章主要介紹了Spring如何利用@Value注解讀取yml中的map配置,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-02-02

最新評(píng)論