SpringBoot整合quartz实现动态启动,停止定时任务功能
2024-10-19 01:04:22
注意:这个方法当程序重启之后会失效,所以必须将定时任务持久化到数据库,然后程序启动的时候重新把数据库的定时任务加载到quartz中
springboot程序启动初始化代码参考:https://www.cnblogs.com/pxblog/p/14995261.html
引入maven
<!-- https://mvnrepository.com/artifact/org.quartz-scheduler/quartz -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-quartz</artifactId>
</dependency>
TaskJob.java 任务执行类
package com.test.cms.task; import com.test.cms.service.TaskService;
import org.quartz.*;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component; import java.util.Date; /**
* @PersistJobDataAfterExecution 和 @DisallowConcurrentExecution
* 表示不让某个定时任务并发执行保证上一个任务执行完后,再去执行下一个任务
*/
@PersistJobDataAfterExecution
@DisallowConcurrentExecution
@Component
public class TaskJob implements Job { /**
* 可以直接注入service层。这里只是演示,没有这个类
*/
@Autowired
private TaskService taskService; @Override
public void execute(JobExecutionContext context) {
JobDataMap jobDataMap=context.getJobDetail().getJobDataMap();
System.out.println("11"+new Date()+" "+jobDataMap.getString("value")+" "+context.getJobDetail().getKey());
} }
控制器
TaskController.java
package com.test.cms.controller; import com.test.cms.task.TaskJob;
import org.quartz.JobDataMap;
import org.quartz.JobKey;
import org.quartz.Scheduler;
import org.quartz.SchedulerException;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.scheduling.quartz.CronTriggerFactoryBean;
import org.springframework.scheduling.quartz.JobDetailFactoryBean;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController; import java.util.*; @RestController
public class TaskController { /**
* 启动任务 需要自己完善逻辑,这里我用uuid作为taskCode 保证唯一
* 启动之前要通过数据库查询是否任务已经启动,如果启动了就不能启动了
* 启动成功了 要把数据库的任务状态改为启动中
*/
@RequestMapping(value = "/task/start")
public void start() {
String uuid = UUID.randomUUID().toString();
System.out.println(uuid);
startTask(uuid); } /**
* 停止任务 需要自己完善逻辑
* @param taskCode 传入启动任务时设置的taskCode参数
*/
@RequestMapping(value = "/task/stop")
public void stop(String taskCode) {
endTask(taskCode);
} /**
* 开始任务调度
*
* @param taskCode 任务名称 需要唯一标识,停止任务时需要用到
*/
private void startTask(String taskCode){
//任务开始的corn表达式
String cronExpress = "*/5 * * * * ?";
if (cronExpress.indexOf("null") == -1) {
try {
JobDetailFactoryBean jobDetailFactoryBean = new JobDetailFactoryBean();
jobDetailFactoryBean.setName(taskCode);
jobDetailFactoryBean.setGroup(Scheduler.DEFAULT_GROUP);
//TaskJob.class 是任务所要执行操作的类
jobDetailFactoryBean.setJobClass(TaskJob.class);
//任务需要的参数可以通过map方法传递,
Map map = new HashMap();
map.put("value", "我在传递参数");
jobDetailFactoryBean.setJobDataMap(getJobDataMap(map));
jobDetailFactoryBean.afterPropertiesSet();
CronTriggerFactoryBean cronTriggerFactoryBean = new CronTriggerFactoryBean();
cronTriggerFactoryBean.setBeanName(taskCode);
cronTriggerFactoryBean.setCronExpression(cronExpress);
cronTriggerFactoryBean.setGroup(Scheduler.DEFAULT_GROUP);
cronTriggerFactoryBean.setName("cron_" + taskCode);
cronTriggerFactoryBean.afterPropertiesSet();
scheduler.scheduleJob(jobDetailFactoryBean.getObject(), cronTriggerFactoryBean.getObject());
System.out.println(taskCode+"任务启动成功");
} catch (Exception e) {
e.printStackTrace();
System.out.println(taskCode+"任务启动失败");
}
}
} /**
* 结束任务调度
*
* @param taskCode
*/
private void endTask(String taskCode) {
try {
scheduler.deleteJob(JobKey.jobKey(taskCode, Scheduler.DEFAULT_GROUP));
System.out.println(taskCode+"任务停止成功");
} catch (SchedulerException e) {
e.printStackTrace();
System.out.println(taskCode+"任务停止失败");
}
} /**
* 将HashMap转为JobDataMap
* @param params
* @return
*/
private JobDataMap getJobDataMap(Map<String, String> params) {
JobDataMap jdm = new JobDataMap();
Set<String> keySet = params.keySet();
Iterator<String> it = keySet.iterator();
while (it.hasNext()) {
String key = it.next();
jdm.put(key, params.get(key));
}
return jdm;
} @Autowired
private Scheduler scheduler;
}
启动类要加上注解@EnableScheduling
package com.test.cms; import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.scheduling.annotation.EnableScheduling; @SpringBootApplication
@EnableScheduling //要加上开启定时任务的注解
public class CmsApplication { public static void main(String[] args) {
SpringApplication.run(CmsApplication.class, args);
} }
JAVA日期Date格式转corn表达式:https://www.cnblogs.com/pxblog/p/14992923.html
最新文章
- Caffe源码解析3:Layer
- H5 学习笔记(一、关于position定位)
- 如何解决requireJs的模块加载超时
- [原创] hadoop学习笔记:卸载和安装jdk
- SharePoint 2010 PowerShell 系列 之 备份、还原、部署 .WSP
- Memo打印1
- POJ 2230 (欧拉路)
- js里面“===”与“==”的区别
- Android短信监听(二)——利用ContentObserver实现短信监听
- PHP获得上(两)周时间
- Mysql查询优化器浅析
- 三位数流水码的生成(000&#183;&#183;&#183;&#183;&#183;009&#183;&#183;00A&#183;&#183;&#183;&#183;00Z&#183;&#183;&#183;&#183;ZZZ)
- 字符串API练习三则
- git命令行工作环境配置【转】
- fPLL结构及动态配置
- HDU 6103 Kirinriki (思维 双指针)
- jconsole监控上Linux上的JVM
- Nagios 监控mysqlserver具体实现过程
- java命令--jmap命令使用(查找内存泄漏对象)
- python学习——单例模式
热门文章
- Codeforces 679E - Bear and Bad Powers of 42(线段树+势能分析)
- Boussinesq 近似及静压假定,内外模分离方法(附录A)
- 蛋白质组DIA深度学习之谱图预测
- Python基础之列表内置方法
- goto 的用法
- Centos7服务器上RabbitMQ单机安装
- MapReduce02 序列化
- accent, access, accident
- flink-----实时项目---day05-------1. ProcessFunction 2. apply对窗口进行全量聚合 3使用aggregate方法实现增量聚合 4.使用ProcessFunction结合定时器实现排序
- 大数据学习day23-----spark06--------1. Spark执行流程(知识补充:RDD的依赖关系)2. Repartition和coalesce算子的区别 3.触发多次actions时,速度不一样 4. RDD的深入理解(错误例子,RDD数据是如何获取的)5 购物的相关计算