当前位置: 首页 > 编程日记 > 正文

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

版权声明:本文为博主原创文章,转载请注明出处。 https://blog.csdn.net/liuchuanhong1/article/details/60873295

前言

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

一、新建一个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 {@Id@GeneratedValue(strategy = GenerationType.AUTO)private Long id;@Columnprivate 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的namereturn tigger;}/*** attention:* Details:定义quartz调度工厂*/@Bean(name = "scheduler")public SchedulerFactoryBean schedulerFactory(Trigger cronJobTrigger) {SchedulerFactoryBean bean = new SchedulerFactoryBean();// 用于quartz集群,QuartzScheduler 启动时更新己存在的Jobbean.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;@Configuration
@EnableScheduling
@Component
public class ScheduleRefreshDatabase {@Autowiredprivate 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表达式重新构建triggertrigger = (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推荐的方式完美,但基本上可以满足我们的需求,当然也可以采用触发事件的方式来实现,例如当前端修改定时任务的触发时间时,异步的向后台发送通知,后台收到通知后,然后再更新程序,也可以实现动态的定时任务刷新

转载于:https://www.cnblogs.com/tinyj/p/9798456.html

相关文章:

SQLserver 常用函数适用方法(转载)

SQL Server 常用函数使用方法(持续更新) 之前就想要把一些 SQL 的常用函数记录下来&#xff0c;不过一直没有实行。。。嘿嘿。。。 直到今天用到substring()这个函数&#xff0c;C# 里面这个方法起始值是 0&#xff0c;而 SQL 里面起始值是 1。傻傻分不清楚。。。 这篇博客作为…

“接口”的定义及其与“抽象类”的区别

我们知道一个有抽象方法的类是抽象类&#xff0c;而当一个类中全是抽象方法时&#xff0c;就可以定义为接口&#xff08;interface&#xff09; 接口命名通常以“I”开头&#xff1b;接口中的方法默认有public abstract&#xff08;所以可以省略&#xff09;&#xff1b;接口中…

Linux13-计划任务crontab

目录 一、用户计划任务 1.1、定义用户计划任务的命令crontab 1.2、作业格式 二、系统计划任务cron 三、管理临时文件 3.1、systemd-tmpfiles命令与配置文件 3.2、用法举例 一、用户计划任务 1.1、定义用户计划任务的命令crontab Linux提供了针对周期性作业的crond守护…

Java线程安全 关于原子性与volatile的试验

1. 变量递增试验 1 static /*volatile*/ int shared0;//volatile也无法保证操作的原子性2 static synchronized int incrShared(){//不加synchronized的话&#xff0c;shared最终结果值小于预期3 return shared;4 }5 public static void testIncrShare…

Python学习 day01打卡

1.Python : 是一门解释型 弱类型 高级开发编程语言. 2.第一个Python程序的编写: print ("hell,world") 3.变量:把程序运行过程中的值储存起来,方便后面的程序调用. 4.变量的命名要求: 1,只能是数字或字母或下划线组成; 2,不能是数字开头,也不能全是数字; 3,不能是Pyt…

Window10安装pycocotools

之前在linux系统下安装pycocotools时非常简单&#xff0c;只需&#xff1a; pip install pycocotools 如果在windows下通过以上指令安装会报错&#xff0c;网上查询后都说需要安装C编译器或者VS环境&#xff0c;等等&#xff0c;其实没必要那么麻烦&#xff0c;直接通过一下指令…

有名内部类 匿名内部类

内部类&#xff0c;从名字上就可以看出&#xff0c;是定义在外部类里的类&#xff0c;可以更加详细地说明某个事物的特征&#xff0c;举个例子&#xff0c;就像一个人&#xff08;把他定义为一个类[外部类]&#xff09;&#xff0c;这个人有大脑、肝脏、心脏等器官&#xff0c;…

Linux14-进程的优先级nice

目录 一、进程优先级和nice的概念 二、显示nice级别 三、更改nice级别 一、进程优先级和nice的概念 CPU管理进程的是按照时间片的方式来划分CPU的资源。CPU可以把1s划分成若干份&#xff0c;轮询时间片。通过设置进程的优先级&#xff0c;让某些重要的进程优先并且长时间获…

ConcurrentHashMap实现原理及源码分析

ConcurrentHashMap是Java并发包中提供的一个线程安全且高效的HashMap实现&#xff08;若对HashMap的实现原理还不甚了解&#xff0c;可参考我的另一篇文章HashMap实现原理及源码分析&#xff09;&#xff0c;ConcurrentHashMap在并发编程的场景中使用频率非常之高&#xff0c;本…

Gulp的简单使用

我比较喜欢使用Gulp&#xff0c;因为简单好用&#xff01; 今天的任务是&#xff1a;使用Gulp来压缩 jQuery源码&#xff0c;各输出一个压缩的和未压缩的版本 第一步&#xff1a;安装 cnpm install gulp --save-dev cnpm install gulp-rename gulp-uglify --save-dev 第二步&…

Mask_RCNN安装与踩过的坑

一、Mask_RCNN下载 https://www.bilibili.com/video/BV1M7411x7is?t629&p5 按照上述教程的话&#xff0c;安装的是ballon例子的Mask_RCNN。这个理解的话&#xff0c;修改Mask_RCNN也就很简单了。 1.2 更新驱动 强烈建议更新驱动&#xff0c;因为以后如果装pytorch的话&…

Lambda表达式的运用(详尽版)

Lambda表达式是Java JDK8发布的最新特性&#xff0c;它极大地简化了定义匿名内部类时要写的代码&#xff0c;使代码变得更加简洁紧凑。 我们知道定义匿名内部类时要紧跟创建对象&#xff0c;接着要实现接口或重写父类中的抽象方法&#xff0c;例如&#xff1a; public interfa…

Linux15-SELinux

目录 一、查看和配置SELinux生效 二、SELinux上下文 2.1、查看SELinux上下文 2.2、semanagerestorecon命令更改SELinux上下文 三、SELinux布尔值 四、SELinux故障排除 Security Enhanced Linux&#xff08;SELinux&#xff09;是一个额外的系统安全层&#xff0c;是一种强…

Java学习(三)JSP学习1

一、 理解JSP技术 JSP全名为Java Server Pages&#xff0c;中文名叫java服务器页面&#xff0c;其根本是一个简化的Servlet设计&#xff0c;它 是由Sun Microsystems公司倡导、许多公司参与一起建立的一种动态网页技术标准。JSP技术有点类似ASP技术&#xff0c;它是在传统的网…

通过Java代码实现图片的放大和缩小

本文介绍的例子在Android安卓手机上测试通过。 先看看效果吧。可以看到这个开发好的安卓应用有三个按钮&#xff1a;Zoom In缩小图片&#xff0c;Zoom Out放大图片和Save保存。 初始页面&#xff1a; 可以在左边边框自由移动图片&#xff0c;图片下面的调试界面debug screen会显…

ProxyError: Conda cannot proceed due to an error in your proxy configuration

右键网络---->属性---->左下角的Internet选项---->链接---->局域网设置 将所以的对号去掉

Java访问权限(详尽版)

Java中的访问权限一共有四种&#xff1a;public 、protected 、 默认的 、private&#xff08;访问范围由大到小&#xff09;&#xff1a; 先说public和private这两种&#xff0c;因为这两种最容易理解&#xff1a; public&#xff1a;在整个项目中都可以被访问&#xff0c;无…

suse11sp4配置vnc显示gnome

目录 一、suse11sp4配置vnc默认显示xterm 二、suse11sp4配置vnc显示gnome桌面 一、suse11sp4配置vnc默认显示xterm 今天给一台比较老的服务器配置vnc服务&#xff0c;操作系统为suse11sp4。该服务器已经安装了gnome界面和vnc。vncpasswd设置密码&#xff0c;vncserver :1设置…

ubuntu18.04安装CUDA10.0

1、更新驱动 对于CUDA10.0&#xff0c;要求驱动版本大于410.48&#xff0c;因此首先要更新驱动 ubuntu-drivers devices&#xff08;查看所有驱动&#xff09; sudo ubuntu-drivers autoinstall&#xff08;自动安装最适合的驱动&#xff09; 查看驱动版本 nvidia-smi 2、下载…

运行时异常 检查时异常

Throwable是Java程序中所有异常对象的根基类&#xff0c;而Throwable是从Object类直接继承来的&#xff0c; Throwable可分为error&#xff08;错误&#xff09;和 Exception&#xff08;异常&#xff09;。error表示的是JDK出了问题&#xff0c;与写的代码无关&#xff1b;而E…

Atitit. Attilax企业框架 AEF的发展里程总结

Atitit. Attilax企业框架 AEF的发展里程总结 1. Attilax企业框架and框架发展思想 1 2. AEF框架 2 2.1. 多语言支持&#xff0c;涉及的语言 java ,c# php python js c 2 2.2. 业务模块支持 2 2.3. 技术组件支持 2 2.4. 平台化支持 2 2.5. 插件化支持 2 3. AEF框架发展里程 3 3.1…

1.3创建项目「深入浅出ASP.NET Core系列」

控制台创建项目 dotnet new --help 使用控制台采集项目&#xff0c;务必要熟练使用命令&#xff0c;--help是命令帮助的指明灯&#xff0c;在你无法google的时候&#xff0c;可以离线状态最快的帮助到你。 根据模板名称&#xff0c;我们可以创建各种模板&#xff0c;比如 dotne…

Linux16-防火墙与firewalld

目录 一、netfilter与firewalld 二、firewalld与firewall-cmd、firewall-config 2.1、预定义区域 2.2、预定义服务 2.3、firewall-cmd和firewall-config配置工具 2.4、富规则rich-rule 一、netfilter与firewalld linux内核有个防火墙模块netfilter&#xff0c;它是一个 …

throws throw 自定义异常

我们知道当程序遇到异常时除了用try-catch-finally来捕获异常外&#xff0c;还可以用throws和throw去抛出异常&#xff0c;使用异常处理可以定位问题所在处&#xff0c;方便修改程序代码&#xff1a; 例如利用try-catch-finally来捕获异常&#xff1a; public class Test{pub…

配置hadoop集群一

花了1天时间最终把环境搭建好了。整理了一下&#xff0c;希望对想学习hadoop的有所帮助。资料下载&#xff1a;http://pan.baidu.com/s/1kTupgkn包括了linux虚拟机。jdk, hadoop1.0环境搭建准备工作&#xff1a;hadoop_master 192.168.50.158hadoop_slave 192.168.50.136网络…

将前台日期格式转成与数据库日期格式相对应,后台java转数据库日期格式

前台到后台java时data日期类型的转化 在实体类中用DataTimeFormat&#xff0c;这样设置即使传过来是空的字符串也是可以转的&#xff0c;要和前面传过来的格式一致&#xff0c;如 XmlElement(name"BeginDate") DateTimeFormat(pattern"yyyy-MM-dd") privat…

Ubuntu18.04安装cudnn

下载Cudnn cudnn下载版本要与cudn版本相对应。笔者cudn为10.0&#xff0c;则cudnn选7.65。下载地址&#xff1a; https://developer.nvidia.com/rdp/cudnn-archive 下载for Linux这个。 编译 解压后运行以下命令&#xff1a; sudo cp cuda/include/cudnn.h /usr/local/cuda…

Linux17-磁盘分区、文件系统、逻辑卷管理LVM

目录 一、磁盘分区、文件系统、永久挂载 1.1、MBR分区方案 1.2、使用fdisk、mkfs、partprobe、blkid、lsblk等命令管理MBR分区 1.3、swap分区 1.4、GPT分区方案、使用gdisk管理GPT分区 二、逻辑卷LVM&#xff08;Logical Volume Management&#xff09; 2.1、物理卷PV、…

javascript基础 之 json

1&#xff0c;json是用于存储和传输的数据格式 全称&#xff1a;JSON 英文全称 JavaScript Object Notation json转化为javascript的规则&#xff1a; 数据为 键/值 对。数据由逗号分隔。大括号保存对象方括号保存数组<body> <p id"hehe"></p> <…

List常用方法总结 遍历集合的方法

List接口继承Collection接口&#xff0c;该接口属于数据结构中的线性结构&#xff0c;用户可以根据元素的整数索引来访问元素&#xff0c;换句话说就是List集合是按照存储的顺序保存的&#xff0c;且从0开始数&#xff0c;说白了就是就相当于一个数组&#xff0c;不同的是数组要…