激情久久久_欧美视频区_成人av免费_不卡视频一二三区_欧美精品在欧美一区二区少妇_欧美一区二区三区的

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術|正則表達式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

香港云服务器
服務器之家 - 編程語言 - Java教程 - springboot整合Quartz實現動態(tài)配置定時任務的方法

springboot整合Quartz實現動態(tài)配置定時任務的方法

2021-01-18 09:27牛奮lch Java教程

本篇文章主要介紹了springboot整合Quartz實現動態(tài)配置定時任務的方法,非常具有實用價值,需要的朋友可以參考下

前言

在我們日常的開發(fā)中,很多時候,定時任務都不是寫死的,而是寫到數據庫中,從而實現定時任務的動態(tài)配置,下面就通過一個簡單的示例,來實現這個功能。

一、新建一個springboot工程,并添加依賴

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
<dependency>
      <groupId>org.springframework.boot</groupId>
      <artifactId>spring-boot-starter-data-jpa</artifactId>
    </dependency>
 
    <dependency><!-- 為了方便測試,此處使用了內存數據庫 -->
      <groupId>com.h2database</groupId>
      <artifactId>h2</artifactId>
      <scope>runtime</scope>
    </dependency>
    <dependency>
      <groupId>org.springframework.boot</groupId>
      <artifactId>spring-boot-starter-test</artifactId>
      <scope>test</scope>
    </dependency>
     
    <dependency>
      <groupId>org.quartz-scheduler</groupId>
      <artifactId>quartz</artifactId>
      <version>2.2.1</version>
      <exclusions>
        <exclusion>
          <artifactId>slf4j-api</artifactId>
          <groupId>org.slf4j</groupId>
        </exclusion>
      </exclusions>
    </dependency>
    <dependency><!-- 該依賴必加,里面有sping對schedule的支持 -->
            <groupId>org.springframework</groupId>
            <artifactId>spring-context-support</artifactId>
    </dependency>

二、配置文件application.properties

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
# 服務器端口號 
server.port=7902
# 是否生成ddl語句 
spring.jpa.generate-ddl=false 
# 是否打印sql語句 
spring.jpa.show-sql=true 
# 自動生成ddl,由于指定了具體的ddl,此處設置為none 
spring.jpa.hibernate.ddl-auto=none 
# 使用H2數據庫 
spring.datasource.platform=h2 
# 指定生成數據庫的schema文件位置 
spring.datasource.schema=classpath:schema.sql 
# 指定插入數據庫語句的腳本位置 
spring.datasource.data=classpath:data.sql 
# 配置日志打印信息 
logging.level.root=INFO 
logging.level.org.hibernate=INFO 
logging.level.org.hibernate.type.descriptor.sql.BasicBinder=TRACE 
logging.level.org.hibernate.type.descriptor.sql.BasicExtractor=TRACE 
logging.level.com.itmuch=DEBUG 

三、Entity類

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
package com.chhliu.springboot.quartz.entity;
 
import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
 
@Entity
public class Config {
  @Id
   @GeneratedValue(strategy = GenerationType.AUTO)
   private Long id;
 
   @Column
   private String cron;
 
  /**
   * @return the id
   */
  public Long getId() {
    return id;
  }
    ……此處省略getter和setter方法……
}

四、任務類

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
package com.chhliu.springboot.quartz.entity;
 
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.context.annotation.Configuration;
import org.springframework.scheduling.annotation.EnableScheduling;
import org.springframework.stereotype.Component;
 
@Configuration
@Component // 此注解必加
@EnableScheduling // 此注解必加
public class ScheduleTask {
  private static final Logger LOGGER = LoggerFactory.getLogger(ScheduleTask.class);
  public void sayHello(){
    LOGGER.info("Hello world, i'm the king of the world!!!");
  }
}

五、Quartz配置類

由于springboot追求零xml配置,所以下面會以配置Bean的方式來實現

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
package com.chhliu.springboot.quartz.entity;
 
import org.quartz.Trigger;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.scheduling.quartz.CronTriggerFactoryBean;
import org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean;
import org.springframework.scheduling.quartz.SchedulerFactoryBean;
 
@Configuration
public class QuartzConfigration {
  /**
   * attention:
   * Details:配置定時任務
   */
  @Bean(name = "jobDetail")
  public MethodInvokingJobDetailFactoryBean detailFactoryBean(ScheduleTask task) {// ScheduleTask為需要執(zhí)行的任務
    MethodInvokingJobDetailFactoryBean jobDetail = new MethodInvokingJobDetailFactoryBean();
    /*
     * 是否并發(fā)執(zhí)行
     * 例如每5s執(zhí)行一次任務,但是當前任務還沒有執(zhí)行完,就已經過了5s了,
     * 如果此處為true,則下一個任務會執(zhí)行,如果此處為false,則下一個任務會等待上一個任務執(zhí)行完后,再開始執(zhí)行
     */
    jobDetail.setConcurrent(false);
     
    jobDetail.setName("srd-chhliu");// 設置任務的名字
    jobDetail.setGroup("srd");// 設置任務的分組,這些屬性都可以存儲在數據庫中,在多任務的時候使用
     
    /*
     * 為需要執(zhí)行的實體類對應的對象
     */
    jobDetail.setTargetObject(task);
     
    /*
     * sayHello為需要執(zhí)行的方法
     * 通過這幾個配置,告訴JobDetailFactoryBean我們需要執(zhí)行定時執(zhí)行ScheduleTask類中的sayHello方法
     */
    jobDetail.setTargetMethod("sayHello");
    return jobDetail;
  }
   
  /**
   * attention:
   * Details:配置定時任務的觸發(fā)器,也就是什么時候觸發(fā)執(zhí)行定時任務
   */
  @Bean(name = "jobTrigger")
  public CronTriggerFactoryBean cronJobTrigger(MethodInvokingJobDetailFactoryBean jobDetail) {
    CronTriggerFactoryBean tigger = new CronTriggerFactoryBean();
    tigger.setJobDetail(jobDetail.getObject());
    tigger.setCronExpression("0 30 20 * * ?");// 初始時的cron表達式
    tigger.setName("srd-chhliu");// trigger的name
    return tigger;
 
  }
 
  /**
   * attention:
   * Details:定義quartz調度工廠
   */
  @Bean(name = "scheduler")
  public SchedulerFactoryBean schedulerFactory(Trigger cronJobTrigger) {
    SchedulerFactoryBean bean = new SchedulerFactoryBean();
    // 用于quartz集群,QuartzScheduler 啟動時更新己存在的Job
    bean.setOverwriteExistingJobs(true);
    // 延時啟動,應用啟動1秒后
    bean.setStartupDelay(1);
    // 注冊觸發(fā)器
    bean.setTriggers(cronJobTrigger);
    return bean;
  }
}

六、定時查庫,并更新任務

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
package com.chhliu.springboot.quartz.entity;
 
import javax.annotation.Resource;
 
import org.quartz.CronScheduleBuilder;
import org.quartz.CronTrigger;
import org.quartz.JobDetail;
import org.quartz.Scheduler;
import org.quartz.SchedulerException;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Configuration;
import org.springframework.scheduling.annotation.EnableScheduling;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;
 
import com.chhliu.springboot.quartz.repository.ConfigRepository;
 
@Configuration
@EnableScheduling
@Component
public class ScheduleRefreshDatabase {
  @Autowired
  private ConfigRepository repository;
 
  @Resource(name = "jobDetail")
  private JobDetail jobDetail;
 
  @Resource(name = "jobTrigger")
  private CronTrigger cronTrigger;
 
  @Resource(name = "scheduler")
  private Scheduler scheduler;
 
  @Scheduled(fixedRate = 5000) // 每隔5s查庫,并根據查詢結果決定是否重新設置定時任務
  public void scheduleUpdateCronTrigger() throws SchedulerException {
    CronTrigger trigger = (CronTrigger) scheduler.getTrigger(cronTrigger.getKey());
    String currentCron = trigger.getCronExpression();// 當前Trigger使用的
    String searchCron = repository.findOne(1L).getCron();// 從數據庫查詢出來的
    System.out.println(currentCron);
    System.out.println(searchCron);
    if (currentCron.equals(searchCron)) {
      // 如果當前使用的cron表達式和從數據庫中查詢出來的cron表達式一致,則不刷新任務
    } else {
      // 表達式調度構建器
      CronScheduleBuilder scheduleBuilder = CronScheduleBuilder.cronSchedule(searchCron);
      // 按新的cronExpression表達式重新構建trigger
      trigger = (CronTrigger) scheduler.getTrigger(cronTrigger.getKey());
      trigger = trigger.getTriggerBuilder().withIdentity(cronTrigger.getKey())
          .withSchedule(scheduleBuilder).build();
      // 按新的trigger重新設置job執(zhí)行
      scheduler.rescheduleJob(cronTrigger.getKey(), trigger);
      currentCron = searchCron;
    }
  }
}

六、相關腳本

1、data.sql

?
1
insert into config(id,cron) values(1,'0 0/2 * * * ?'); # 每2分鐘執(zhí)行一次定時任務

2、schema.sql

?
1
2
3
4
5
6
drop table config if exists;
create table config(
  id bigint generated by default as identity,
  cron varchar(40),
  primary key(id)
);

六、運行測試

測試結果如下:(Quartz默認的線程池大小為10)

?
1
2
3
4
5
6
0 30 20 * * ?
0 0/2 * * * ?
2017-03-08 18:02:00.025 INFO 5328 --- [eduler_Worker-1] c.c.s.quartz.entity.ScheduleTask     : Hello world, i'm the king of the world!!!
2017-03-08 18:04:00.003 INFO 5328 --- [eduler_Worker-2] c.c.s.quartz.entity.ScheduleTask     : Hello world, i'm the king of the world!!!
2017-03-08 18:06:00.002 INFO 5328 --- [eduler_Worker-3] c.c.s.quartz.entity.ScheduleTask     : Hello world, i'm the king of the world!!!
2017-03-08 18:08:00.002 INFO 5328 --- [eduler_Worker-4] c.c.s.quartz.entity.ScheduleTask     : Hello world, i'm the king of the world!!!

從上面的日志打印時間來看,我們實現了動態(tài)配置,最初的時候,任務是每天20:30執(zhí)行,后面通過動態(tài)刷新變成了每隔2分鐘執(zhí)行一次。

雖然上面的解決方案沒有使用Quartz推薦的方式完美,但基本上可以滿足我們的需求,當然也可以采用觸發(fā)事件的方式來實現,例如當前端修改定時任務的觸發(fā)時間時,異步的向后臺發(fā)送通知,后臺收到通知后,然后再更新程序,也可以實現動態(tài)的定時任務刷新

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持服務器之家。

原文鏈接:http://blog.csdn.net/liuchuanhong1/article/details/60873295

延伸 · 閱讀

精彩推薦
572
主站蜘蛛池模板: 日韩视频在线一区二区三区 | 精品亚洲午夜久久久久91 | 日本a v免费观看 | 欧美wwwwww| 国产精品久久久久久久模特 | 久久精品国产一区二区电影 | 激情网站免费观看 | 国产高潮好爽好大受不了了 | 日本一级黄色大片 | 精品国产91一区二区三区 | 久久综合精品视频 | 色中色激情影院 | 成人免费一区二区三区 | 一级成人欧美一区在线观看 | 少妇一级淫片免费看 | 国产美女视频免费 | 国产69精品久久久久久久久久 | 欧美高清视频一区 | 九九热免费精品 | 三级18视频 | 蜜桃视频网站在线观看 | 国产精品视频专区 | 中国国语毛片免费观看视频 | 国产资源在线视频 | 久久久婷婷一区二区三区不卡 | 国产免费传媒av片在线 | 国产乱一区二区三区视频 | 日日摸夜夜骑 | 久久久久性 | 欧美国产永久免费看片 | 欧美成人三级视频 | 男女隐私免费视频 | 中文日韩欧美 | 一级黄色免费 | 国产羞羞视频在线免费观看 | 久久久国产一区二区三区 | 亚洲欧美在线视频免费 | 亚洲第一黄色网 | 国产精品自拍av | 国产在线观看福利 | 97人人草 |