赞
踩
目录
定时任务的实现方式有很多,如下:
1.启动类中添加@EnableScheduling,开启定时任务功能,然后通过注解的方式,如
@Scheduled(cron = "0/1 * * * * ?")设置定时任务的执行方式,定时任务要设置成一个spring容器的bean
2.通过第三方框架的方式,如xxl-job,
elastic-Job。
3.通过quartz方式,第三方的框架也是基于quartz实现的。
2.quartz说明
quartz是完全有java开发的作业调度框架,包含一下几个核心概念:
是quartz中的任务调度器,通过trigger和jobdetail进行调度,暂停和删除任务,相当于一个容器,装载着任务和触发器。Trigger 和 JobDetail 可以注册到 Scheduler 中,两者在 Scheduler 中拥有各自的组及名称,组及名称是 Scheduler 查找定位容器中某一对象的依据,Trigger 的组及名称必须唯一,JobDetail 的组和名称也必须唯一(但可以和 Trigger 的组和名称相同,因为它们是不同类型的)。
(2)trigger
Quartz 中的触发器,是一个类,描述触发 Job
执行的时间触发规则,主要有 SimpleTrigger
和 CronTrigger
这两个子类。
(3)jobDetail
Quartz 中需要执行的任务详情,包括了任务的唯一标识和具体要执行的任务,可以通过 JobDataMap
往任务中传递数据。
(4)job
Quartz 中具体的任务,包含了任务的具体实现逻辑。
(5)JobBuilder
用于创建一个任务实例,也可以定义关于该任务的详情比如任务名、组名等,这个声明的实例将会作为一个实际执行的任务。
(6)TriggerBuilder
触发器创建器,用于创建触发器trigger实例。
(7)监听器组件
JobListener、TriggerListener、SchedulerListener监听器,用于对组件的监听。
默认情况下 Quartz 会将任务调度存储在内存中,这种方式性能是最好的,因为内存的速度是最快的。不好的地方就是数据缺乏持久性,但程序崩溃或者重新发布的时候,所有运行信息都会丢失。
(2)数据库方式
存储数据库后,可以做单点也可以做集群,当任务多了之后,可以统一进行管理,随时停止、暂停、修改任务。 关闭或者重启服务器,运行的信息都不会丢失。缺点就是运行速度快慢取决于连接数据库的快慢。
(1)引入依赖
springboot项目可以直接引入下面依赖。
- <dependency>
- <groupId>org.springframework.boot</groupId>
- <artifactId>spring-boot-starter-quartz</artifactId>
- </dependency>
非springboot项目引入下面依赖,当然springboot项目也可以使用此依赖
- <dependency>
- <groupId>org.quartz-scheduler</groupId>
- <artifactId>quartz</artifactId>
- <version>2.2.3</version>
- </dependency>
(2)开发定时任务
- package com.example.demo.job;
-
- import org.quartz.JobExecutionContext;
- import org.quartz.JobExecutionException;
- import org.springframework.scheduling.quartz.QuartzJobBean;
-
- /**
- * @Author linaibo
- * @Date 2023/11/17 16:52
- * @Version 1.0
- */
- public class SimpleJob extends QuartzJobBean {
- @Override
- protected void executeInternal(JobExecutionContext jobExecutionContext) throws JobExecutionException {
- System.out.println("执行成功");
- }
- }
定时任务继承QuartzJobBean,并实现其方法。
(3)配置定时任务
- package com.example.demo.config;
-
- import org.quartz.*;
- import org.springframework.beans.factory.annotation.Autowired;
- import org.springframework.boot.ApplicationArguments;
- import org.springframework.boot.ApplicationRunner;
- import org.springframework.context.annotation.Configuration;
- import org.springframework.core.annotation.Order;
-
- /**
- * @Author linaibo
- * @Date 2023/11/17 16:54
- * @Version 1.0
- */
- @Configuration
- @Order(2)
- public class QuartzConfig implements ApplicationRunner {
- @Autowired
- private Scheduler scheduler;
-
- @Override
- public void run(ApplicationArguments args) throws Exception {
- try{
- String cron = "0/5 0/1 * * * ?";
- Class<? extends Job> jobClazz = (Class<? extends Job>)Class.forName("com.example.demo.job.SimpleJob");
- JobDetail jobDetail = JobBuilder.newJob(jobClazz)
- .withIdentity("test")
- .storeDurably()
- .build();
-
- CronTrigger cronTrigger = TriggerBuilder.newTrigger()
- .withIdentity("test")
- .startNow()
- .withSchedule(CronScheduleBuilder.cronSchedule(cron))
- .build();
-
- scheduler.scheduleJob(jobDetail, cronTrigger);
- } catch (ClassNotFoundException e) {
- System.out.println("定时任务类路径出错:请输入类的绝对路径");
- } catch (SchedulerException e) {
- System.out.println("创建定时任务出错");
- }
-
- }
- }
这里使用ApplicationRunner,简单说明一下:
ApplicationRunner常用于项目启动后,(也就是ApringApplication.run()执行结束),立马执行某些逻辑。可用于项目的准备工作,比如加载配置文件,加载执行流,定时任务等等。
使用方式:
实现ApplicationRunner接口,重写run方法,定义具体的执行逻辑
@Order注解,用于决定多个bean的执行顺序,按照值从小到大执行 (值可为负数)
@Order(-1)优先于@Order(0)
@Order(1)优先于@Order(2)
还有个接口,也可以实现和ApplicationRunner一样的功能
CommandLineRunner接口的run方法接收的参数为String数组
创建配置文件实现ApplicationRunner接口,实现run方法,在方法中,首先通过反射方式获取到要执行定时任务的类来创建jobdetail,然后根据执行规则创建执行的触发器trigger,再将jobdetail和trigger放到scheduler容器中,按照规则进行任务的执行。
定时任务的全路径名称,执行的触发器以及任务名称,可以存放在数据库中,并在前端追加可视化画面来进行修改,删除及立即执行的功能。
定时任务的数据表可以自己设计,也可以参照官网,如下
初始化 Quartz 数据表
下载 Quartz 发布包:Downloads
解压缩进入SQL脚本所在位置:quartz-2.3.0-SNAPSHOT/src/org/quartz/impl/jdbcjobstore/tables_mysql_innodb.sql
导入 tables_mysql_innodb.sql 脚本文件。
(1)重新定制
- @Service
- public class QuartzJobServiceImpl implements QuartzJobService {
-
- @Autowired
- private Scheduler scheduler;
-
- @SneakyThrows
- @Override
- public void rescheduleJob(String jobName, String jobGroupName, String cron) {
- TriggerKey triggerKey = TriggerKey.triggerKey(jobName, jobGroupName);
- CronTrigger cronTrigger = TriggerBuilder.newTrigger()
- .withIdentity(triggerKey)
- .startNow()
- .withSchedule(CronScheduleBuilder.cronSchedule(cron))
- .build();
- scheduler.rescheduleJob(triggerKey, cronTrigger);
- }
(2)恢复
- @Service
- public class QuartzJobServiceImpl implements QuartzJobService {
-
- @Autowired
- private Scheduler scheduler;
-
- @SneakyThrows
- @Override
- public void resumeJob(String jobName, String jobGroupName) {
- scheduler.resumeJob(JobKey.jobKey(jobName, jobGroupName));
- }
- }
(3)暂停
- @Service
- public class QuartzJobServiceImpl implements QuartzJobService {
-
- @Autowired
- private Scheduler scheduler;
-
- @SneakyThrows
- @Override
- public void pauseJob(String jobName, String jobGroupName) {
- scheduler.pauseJob(JobKey.jobKey(jobName, jobGroupName));
- }
- }
(4)删除
- @Service
- public class QuartzJobServiceImpl implements QuartzJobService {
-
- @Autowired
- private Scheduler scheduler;
-
- @SneakyThrows
- @Override
- public void deleteJob(String jobName, String jobGroupName) {
- scheduler.pauseTrigger(TriggerKey.triggerKey(jobName, jobGroupName));
- scheduler.unscheduleJob(TriggerKey.triggerKey(jobName, jobGroupName));
- scheduler.deleteJob(JobKey.jobKey(jobName, jobGroupName));
- }
- }
参照:
Spring Boot 集成 Quartz(任务调度框架)_springboot集成quartz_人人都在发奋的博客-CSDN博客
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。