百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 编程网 > 正文

Spring Boot整合Quartz如何定义动态的Job?

yuyutoo 2024-11-01 15:55 4 浏览 0 评论

在Quartz中想要实现动态任务定义,可以通过创建任务的时候的动态灵活参数的方式来实现对于调用策略动态配置从而实现了动态执行任务的操作。下面我们就来介绍一下SpringBoot整合Quartz实现动态Job的定义。

添加依赖

在POM文件中添加Quartz和Spring Boot的依赖,如下所示。

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-quartz</artifactId>
</dependency>

配置Quartz Scheduler

接下来,需要定义一个QuartzSchedulerConfig配置类,用来对Quartz Scheduler进行初始化操作,如下所示。

import org.quartz.Scheduler;
import org.quartz.spi.JobFactory;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.scheduling.quartz.SchedulerFactoryBean;

@Configuration
public class QuartzSchedulerConfig {

    @Bean
    public Scheduler scheduler(JobFactory jobFactory) throws Exception {
        SchedulerFactoryBean factory = new SchedulerFactoryBean();
        factory.setJobFactory(jobFactory);
        factory.afterPropertiesSet();
        Scheduler scheduler = factory.getScheduler();
        scheduler.start();
        return scheduler;
    }
}

定义动态Job类

继承Job接口定一个用来执行动态任务的DynamicJob类,代码如下所示。

import org.quartz.Job;
import org.quartz.JobExecutionContext;
import org.quartz.JobExecutionException;
import org.springframework.stereotype.Component;

@Component
public class DynamicJob implements Job {

    @Override
    public void execute(JobExecutionContext context) throws JobExecutionException {
        String jobParam = (String) context.getJobDetail().getJobDataMap().get("param");
        System.out.println("Dynamic Job executed with param: " + jobParam);
    }
}

定义动态Job服务类

定义一个用来处理动态Job的服务类,并且要确保DynamicJobService类有一个方法来处理任务调度,如下所示。

import org.quartz.*;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;

@Service
public class DynamicJobService {

    @Autowired
    private Scheduler scheduler;

    public void scheduleDynamicJob(String jobName, String jobParam, String cronExpression) throws SchedulerException {
        JobDataMap jobDataMap = new JobDataMap();
        jobDataMap.put("param", jobParam);

        JobDetail jobDetail = JobBuilder.newJob(DynamicJob.class)
                .withIdentity(jobName)
                .usingJobData(jobDataMap)
                .build();

        Trigger trigger = TriggerBuilder.newTrigger()
                .forJob(jobDetail)
                .withIdentity(jobName + "Trigger")
                .withSchedule(CronScheduleBuilder.cronSchedule(cronExpression))
                .build();

        scheduler.scheduleJob(jobDetail, trigger);
    }
}

控制层代码实现

接下来就需要创建RESTFul接口调用的控制层类,如下所示。

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.*;

@RestController
@RequestMapping("/api/jobs")
public class JobController {

    @Autowired
    private DynamicJobService dynamicJobService;

    @PostMapping("/schedule")
    public String scheduleJob(
            @RequestParam String jobName,
            @RequestParam String jobParam,
            @RequestParam String cronExpression) {
        try {
            dynamicJobService.scheduleDynamicJob(jobName, jobParam, cronExpression);
            return "Job scheduled successfully";
        } catch (Exception e) {
            return "Error scheduling job: " + e.getMessage();
        }
    }
}

创建完成之后,接下来我们就可以启动项目然后调用http://localhost:8080/api/jobs/schedule接口,并且这个接口接收了三个参数。

  • jobName: 任意任务名称,例如 "dynamicJob1"
  • jobParam: 动态参数,例如 "Dynamic Param 1"
  • cronExpression: Cron表达式,例如 "0/30 * * * * ?"

如下所示,可以发送如下的请求。

curl -X POST "http://localhost:8080/api/jobs/schedule" \
    -d "jobName=dynamicJob1" \
    -d "jobParam=Dynamic Param 1" \
    -d "cronExpression=0/30 * * * * ?"

这个请求将每30秒调度一次任务,执行 DynamicJob 的 execute 方法,并打印传递的参数 "Dynamic Param 1"。

总结

这样我们就实现了一个动态Job的发布和管理。当客户端调用了参数并且传递请求参数 jobName, jobParam, 和cronExpression传递给DynamicJobService的scheduleDynamicJob 方法之后。

在scheduleDynamicJob方法中会创建一个JobDataMap实例,并将请求参数jobParam 存储在其中,接下来创建一个JobDetail实例,并且指定DynamicJob作为逻辑处理实现类,然后通过JobDataMap来传递一个参数。

接下来就是创建了一个Trigger实例对象,通过cronExpression参数作为定时任务的触发时间,这样Quartz的Scheduler会将JobDetail和Trigger注册到调度器中。

当cronExpression参数指定的时间到了之后,就会触发任务调度,每当触发任务调度的时候,就会触发DynamicJob的execute方法,然后从 JobExecutionContext 中获取任务参数,并执行相应的逻辑。在上面的例子中展示的就是一个简单的参数打印操作。当然在实际使用场景中我们可以添加更加复杂的处理逻辑。

相关推荐

【Socket】解决UDP丢包问题

一、介绍UDP是一种不可靠的、无连接的、基于数据报的传输层协议。相比于TCP就比较简单,像写信一样,直接打包丢过去,就不用管了,而不用TCP这样的反复确认。所以UDP的优势就是速度快,开销小。但是随之...

深入学习IO多路复用select/poll/epoll实现原理

Linux服务器处理网络请求有三种机制,select、poll、epoll,本文打算深入学习下其实现原理。0.结论...

25-1-Python网络编程-基础概念

1-网络编程基础概念1-1-基本概念1-2-OSI七层网络模型OSI(开放系统互联)七层网络模型是国际标准化组织(ISO)提出的网络通信分层架构,用于描述计算机网络中数据传输的过程。...

Java NIO多路复用机制

NIO多路复用机制JavaNIO(Non-blockingI/O或NewI/O)是Java提供的用于执行非阻塞I/O操作的API,它极大地增强了Java在处理网络通信和文件系统访问方面的能力。N...

Python 网络编程完全指南:从零开始掌握 Socket 和网络工具

Python网络编程完全指南:从零开始掌握Socket和网络工具在现代应用开发中,网络编程是不可或缺的技能。Python提供了一系列高效的工具和库来处理网络通信、数据传输和协议操作。本指南将从...

Rust中的UDP编程:高效网络通信的实践指南

在实时性要求高、允许少量数据丢失的场景中,UDP(用户数据报协议)凭借其无连接、低延迟的特性成为理想选择。Rust语言凭借内存安全和高性能的特点,为UDP网络编程提供了强大的工具支持。本文将深入探讨如...

Python 网络编程的基础复习:理解Socket的作用

计算机网络的组成部分在逻辑上可以划分为这样的结构五层网络体系应用层:应用层是网络协议的最高层,解决的是具体应用问题...

25-2-Python网络编程-TCP 编程示例

2-TCP编程示例应用程序通常通过“套接字”(socket)向网络发出请求或者应答网络请求,使主机间或者一台计算机上的进程间可以通信。Python语言提供了两种访问网络服务的功能。...

linux下C++ socket网络编程——即时通信系统(含源码)

一:项目内容本项目使用C++实现一个具备服务器端和客户端即时通信且具有私聊功能的聊天室。目的是学习C++网络开发的基本概念,同时也可以熟悉下Linux下的C++程序编译和简单MakeFile编写二:需...

Python快速入门教程7:循环语句

一、循环语句简介循环语句用于重复执行一段代码块,直到满足特定条件为止。Python支持两种主要的循环结构:for循环和while循环。...

10分钟学会Socket通讯,学不会你打我

Socket通讯是软硬件直接常用的一种通讯方式,分为TCP和UDP通讯。在我的职业生涯中,有且仅用过一次UDP通讯。而TCP通讯系统却经常写,正好今天写了一个TCP通讯的软件。总结一下内容软件使用C#...

Python 高级编程之网络编程 Socket(六)

一、概述Python网络编程是指使用Python语言编写的网络应用程序。这种编程涉及到网络通信、套接字编程、协议解析等多种方面的知识。...

linux网络编程Socket之RST详解

产生RST的三个条件:1.目的地为某端口的SYN到达,然而该端口上没有正在监听的服务器;2.TCP想取消一个已有的连接;3.TCP接收到一个根本不存在的连接上的分节;现在模拟上面的三种情况:cl...

ABB机器人编程实用技巧,多项案例

...

Python中实现Socket通讯(附详细代码)

套接字(socket)是一种在计算机网络中进行进程间通信的方法,它允许不同主机上的程序通过网络相互通信。套接字是网络编程的基础,几乎所有的网络应用程序都使用某种形式的套接字来实现网络功能。套接字可以用...

取消回复欢迎 发表评论: