网易首页 > 网易号 > 正文 申请入驻

springboot整合Quartz实现动态配置定时任务

0
分享至

前言

在我们日常的开发中,很多时候,定时任务都不是写死的,而是写到数据库中,从而实现定时任务的动态配置,下面就通过一个简单的示例,来实现这个功能。

一、新建一个springboot工程,并添加依赖

<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

服务器端口号
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类

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 {br/>@Id<br @GeneratedValue(strategy = GenerationType.AUTO)
private Long id;

@Column private String cron;/** * @return the id */public Long getId() { return id; } ……此处省略getter和setter方法……

}

四、任务类

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的方式来实现

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为需要执行的任务
    MethodInvokingJobDetailFactoryBean jobDetail = new MethodInvokingJobDetailFactoryBean();
    /

    • 是否并发执行
    • 例如每5s执行一次任务,但是当前任务还没有执行完,就已经过了5s了,
    • 如果此处为true,则下一个任务会执行,如果此处为false,则下一个任务会等待上一个任务执行完后,再开始执行
      */
      jobDetail.setConcurrent(false);

      jobDetail.setName("srd-chhliu");// 设置任务的名字
      jobDetail.setGroup("srd");// 设置任务的分组,这些属性都可以存储在数据库中,在多任务的时候使用

      /*

    • 为需要执行的实体类对应的对象
      */
      jobDetail.setTargetObject(task);

      /*

    • sayHello为需要执行的方法
    • 通过这几个配置,告诉JobDetailFactoryBean我们需要执行定时执行ScheduleTask类中的sayHello方法
      */
      jobDetail.setTargetMethod("sayHello");
      return jobDetail;
      }

    /**

  • attention:
  • Details:配置定时任务的触发器,也就是什么时候触发执行定时任务
    /
    @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);
    // 注册触发器
    bean.setTriggers(cronJobTrigger);
    return bean;
    }
    }

    六、定时查库,并更新任务

    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;

@Configurationbr/>@EnableScheduling<br @Component
public class ScheduleRefreshDatabase {br/>@Autowired<br 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执行 scheduler.rescheduleJob(cronTrigger.getKey(), trigger); currentCron = searchCron; } }

}

七、相关脚本

1、data.sql

insert into config(id,cron) values(1,'0 0/2 * ?'); # 每2分钟执行一次定时任务
2、schema.sql
drop table config if exists;
create table config(
id bigint generated by default as identity,
cron varchar(40),
primary key(id)
);

八、运行测试

测试结果如下:(Quartz默认的线程池大小为10)

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!!!

总结:

从上面的日志打印时间来看,我们实现了动态配置,最初的时候,任务是每天20:30执行,后面通过动态刷新变成了每隔2分钟执行一次。
虽然上面的解决方案没有使用Quartz推荐的方式完美,但基本上可以满足我们的需求,当然也可以采用触发事件的方式来实现,例如当前端修改定时任务的触发时间时,异步的向后台发送通知,后台收到通知后,然后再更新程序,也可以实现动态的定时任务刷新

特别声明:以上内容(如有图片或视频亦包括在内)为自媒体平台“网易号”用户上传并发布,本平台仅提供信息存储服务。

Notice: The content above (including the pictures and videos if any) is uploaded and posted by a user of NetEase Hao, which is a social media platform and only provides information storage services.

相关推荐
热点推荐
贾跃亭是怎么还掉100亿美元的?

贾跃亭是怎么还掉100亿美元的?

元气科技馆
2024-04-26 15:36:34
中国人口死亡大数据,需引起重视

中国人口死亡大数据,需引起重视

医者真言
2024-04-26 16:21:09
表姐电脑被偷了,我们安慰她:一个破电脑而已,谁知她说…..

表姐电脑被偷了,我们安慰她:一个破电脑而已,谁知她说…..

黄丽搞笑小能手
2024-04-22 07:30:40
他是顶级骗子,掌掴省级官员,拍省委书记桌子,操纵一省官员升迁

他是顶级骗子,掌掴省级官员,拍省委书记桌子,操纵一省官员升迁

极品小牛肉
2024-01-24 10:09:24
卖不掉就往中国扔?这4款车在国外面临停产,在国内却被捧成豪车

卖不掉就往中国扔?这4款车在国外面临停产,在国内却被捧成豪车

户外小阿隋
2024-04-21 07:45:03
是否加入这一军事联盟,日本悄然降低了调门 | 京酿馆

是否加入这一军事联盟,日本悄然降低了调门 | 京酿馆

新京报评论
2024-04-26 14:55:47
亨德利:没有人能阻止奥沙利文世锦赛第8次夺冠 火箭能打到55岁

亨德利:没有人能阻止奥沙利文世锦赛第8次夺冠 火箭能打到55岁

罗克
2024-04-26 21:26:26
小仓优香:一系列作品得到好评,轻松展现完美熟女形象

小仓优香:一系列作品得到好评,轻松展现完美熟女形象

忆史君
2024-04-26 08:37:44
如果康熙传位给他,清朝很可能成为超级大国,八国联军不是威胁!

如果康熙传位给他,清朝很可能成为超级大国,八国联军不是威胁!

玉玉
2023-12-20 20:56:31
公诉人:“别人打你,你为何要还手”?这起案子引起网友的民愤!

公诉人:“别人打你,你为何要还手”?这起案子引起网友的民愤!

辉哥说动漫
2024-04-24 18:49:38
关注民生的煤气一词,竟成为敏感词语,实在想不明白,怎么会这样

关注民生的煤气一词,竟成为敏感词语,实在想不明白,怎么会这样

天闻地知
2024-04-24 14:06:59
佐罗,成功的卖国贼,卖掉了自己国家三分之一的领土

佐罗,成功的卖国贼,卖掉了自己国家三分之一的领土

老土历史
2024-03-29 09:06:11
3天工资到手!凯恩单赛季参与进球超40个 收到25万欧额外奖金

3天工资到手!凯恩单赛季参与进球超40个 收到25万欧额外奖金

直播吧
2024-04-26 18:43:24
徐娇“三角裤事件”升级:还有人翻出她大尺度照片,质疑是在魅男

徐娇“三角裤事件”升级:还有人翻出她大尺度照片,质疑是在魅男

七阿姨爱八卦
2024-03-11 17:05:10
皮奥利:国米过去4年阵容都是意甲最强 但也只拿了两次冠军

皮奥利:国米过去4年阵容都是意甲最强 但也只拿了两次冠军

直播吧
2024-04-26 18:00:27
昆明优化公积金住房套数认定标准:不再将个人住房商贷记录纳入认定范围

昆明优化公积金住房套数认定标准:不再将个人住房商贷记录纳入认定范围

澎湃新闻
2024-04-26 22:26:29
深圳富婆回国邀20名同学聚餐,一顿狂吃8万8,结账时面面相觑

深圳富婆回国邀20名同学聚餐,一顿狂吃8万8,结账时面面相觑

莉雅细细谈
2024-04-06 20:27:48
提灯定损后续,白天人来人往,晚上灯火通明,门口贴着招租

提灯定损后续,白天人来人往,晚上灯火通明,门口贴着招租

普陀动物世界
2024-04-27 00:26:21
广西应急管理厅厅长周长青

广西应急管理厅厅长周长青

探秘桂北
2024-04-27 00:13:22
美女模特开朗直说“你尽管出去找美女,身材比我好比我骚算我输”

美女模特开朗直说“你尽管出去找美女,身材比我好比我骚算我输”

傲娇的马甲线
2024-04-26 17:45:03
2024-04-27 10:14:44
Python技术尖端
Python技术尖端
Java技术能力提升进阶
47文章数 1764关注度
往期回顾 全部

科技要闻

特斯拉这款车型刚上市几天,就上调价格

头条要闻

俄副防长被捕 美媒:或是普京对绍伊古发出的某种信号

头条要闻

俄副防长被捕 美媒:或是普京对绍伊古发出的某种信号

体育要闻

硬不起来的阿波,软不下去的切特

娱乐要闻

金靖回应不官宣恋情结婚的原因

财经要闻

北京房价回到2016年

汽车要闻

2024北京车展 比亚迪的自驱力让对手紧追猛赶

态度原创

家居
艺术
旅游
亲子
军事航空

家居要闻

光影之间 空间暖意打造生活律动

艺术要闻

画廊周北京迎来第八年, “漂留” 主题聚集 30 余家艺术机构与 40 场展览

旅游要闻

散装河北,冀北、冀东、冀中、冀南如何划分?

亲子要闻

小孩带领同学走操场 这气质真是绝了

军事要闻

以军称已完成对拉法地面军事行动准备工作

无障碍浏览 进入关怀版