JAVA可控定时任务(quartz)

目录

功能说明

添加依赖

创建定时任务实体

工具类

service接口

定时任务的实现

具体操作


运行环境jdk1.8+springboot+maven

功能说明

动态的进行定时任务的添加,暂停、修改等。任务信息存储在数据库,所以会涉及到数据库方面的知识

添加依赖

在pom.xml文件中添加依赖

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

创建定时任务实体

我这里使用的是 mybatis-plus ,如果是使用其他的orm框架,理解思路后就好了。



import com.baomidou.mybatisplus.annotation.IdType;
import com.baomidou.mybatisplus.annotation.TableId;
import com.baomidou.mybatisplus.annotation.TableName;

import java.io.Serializable;
import java.util.Date;

@TableName("sys_scheduled")
public class Scheduled implements Serializable {

    private static final long serialVersionUID = 8324997582188610887L;
    @TableId(type = IdType.UUID)
    private String id;
    private String className;
    /**
     * 任务key
     */
    private String jobKey;
    /**
     * 人物的cron表达式
     */
    private String cronExpression;
    private String remark;
    /**
     * 任务状态
     */
    private String status;
    private Date createTime;

    public Date getCreateTime() {
        return createTime;
    }

    public void setCreateTime(Date createTime) {
        this.createTime = createTime;
    }

    public String getId() {
        return id;
    }

    public void setId(String id) {
        this.id = id;
    }

    public String getClassName() {
        return className;
    }

    public void setClassName(String className) {
        this.className = className;
    }

    public String getJobKey() {
        return jobKey;
    }

    public void setJobKey(String jobKey) {
        this.jobKey = jobKey;
    }

    public String getCronExpression() {
        return cronExpression;
    }

    public void setCronExpression(String cronExpression) {
        this.cronExpression = cronExpression;
    }

    public String getRemark() {
        return remark;
    }

    public void setRemark(String remark) {
        this.remark = remark;
    }

    public String getStatus() {
        return status;
    }

    public void setStatus(String status) {
        this.status = status;
    }
}

mapper文件

接口



import com.baomidou.mybatisplus.core.mapper.BaseMapper;
import com.myapp.base.sys.job.entity.Scheduled;
import org.apache.ibatis.annotations.Mapper;
import org.springframework.stereotype.Component;

import java.util.List;

@Mapper
@Component
public interface ScheduledMapper extends BaseMapper<Scheduled> {

    List<Scheduled> selectCanRun();



}

XML

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.myapp.base.sys.job.mapper.ScheduledMapper">

    <resultMap id="scheduled" type="com.myapp.base.sys.job.entity.Scheduled">
        <id column="ID" property="id"></id>
        <result column="CLASS_NAME" property="className"></result>
        <result column="STATUS" property="status"></result>
        <result column="REMARK" property="remark"></result>
        <result column="CRON_EXPRESSION" property="cronExpression"></result>
        <result column="JOB_KEY" property="jobKey"></result>
        <result column="CREATE_TIME" property="createTime"></result>
    </resultMap>

    <select id="selectCanRun" resultMap="scheduled">
        select  * from sys_scheduled where STATUS='1'
    </select>




</mapper>

工具类

主要实现就是在这两个类里面,数据库相关的操作都是增删查改。

QuartzUtils 



import com.myapp.base.sys.job.entity.Scheduled;
import org.quartz.*;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.ApplicationContext;
import org.springframework.stereotype.Component;

import javax.annotation.PostConstruct;

@Component
public class QuartzUtils {

    @Autowired
    private static Scheduler scheduler;

    @Autowired
    ApplicationContext applicationContext;

    @PostConstruct
    public void init() {
        QuartzUtils.scheduler = applicationContext.getBean(Scheduler.class);
    }

    public static void createScheduleJob(Scheduled scheduled) throws Exception{

            //获取到定时任务的执行类  必须是类的绝对路径名称
            //定时任务类需要是job类的具体实现 QuartzJobBean是job的抽象类。
            Class<? extends Job> jobClass = (Class<? extends Job>) Class.forName(scheduled.getClassName());
            // 构建定时任务信息
            JobDetail jobDetail = JobBuilder.newJob(jobClass).withIdentity(scheduled.getJobKey()).build();
            // 设置定时任务执行方式
            CronScheduleBuilder scheduleBuilder = CronScheduleBuilder.cronSchedule(scheduled.getCronExpression());
            // 构建触发器trigger
            CronTrigger trigger = TriggerBuilder.newTrigger().withIdentity(scheduled.getJobKey()).withSchedule(scheduleBuilder).build();
            scheduler.scheduleJob(jobDetail, trigger);

    }

    /**
     * 根据任务名称暂停定时任务
     * @param jobName    定时任务名称
     * @throws SchedulerException
     */
    public static void pauseScheduleJob(String jobName){
        JobKey jobKey = JobKey.jobKey(jobName);
        try {
            scheduler.pauseJob(jobKey);
        } catch (SchedulerException e) {
            System.out.println("暂停定时任务出错:"+e.getMessage());
        }
    }

    /**
     * 根据任务名称恢复定时任务
     * @param jobName    定时任务名称
     * @throws SchedulerException
     */
    public static void resumeScheduleJob(String jobName) {
        JobKey jobKey = JobKey.jobKey(jobName);
        try {
            scheduler.resumeJob(jobKey);
        } catch (SchedulerException e) {
            System.out.println("启动定时任务出错:"+e.getMessage());
        }
    }

    /**
     * 根据任务名称立即运行一次定时任务
     * @param jobName       定时任务名称
     * @throws SchedulerException
     */
    public static void runOnce(String jobName) throws Exception{
        JobKey jobKey = JobKey.jobKey(jobName);
        scheduler.triggerJob(jobKey);
    }

    /**
     * 更新定时任务
     * @param scheduled  定时任务信息类
     * @throws SchedulerException
     */
    public static void updateScheduleJob(Scheduled scheduled)  throws Exception{
            //获取到对应任务的触发器
            TriggerKey triggerKey = TriggerKey.triggerKey(scheduled.getJobKey());
            //设置定时任务执行方式
            CronScheduleBuilder scheduleBuilder = CronScheduleBuilder.cronSchedule(scheduled.getCronExpression());
            //重新构建任务的触发器trigger
            CronTrigger trigger = (CronTrigger) scheduler.getTrigger(triggerKey);
            trigger = trigger.getTriggerBuilder().withIdentity(triggerKey).withSchedule(scheduleBuilder).build();
            //重置对应的job
            scheduler.rescheduleJob(triggerKey, trigger);
    }

    /**
     * 根据定时任务名称从调度器当中删除定时任务
     * @param jobName   定时任务名称
     * @throws SchedulerException
     */
    public static void deleteScheduleJob(String jobName) throws Exception{
        TriggerKey triggerKey = TriggerKey.triggerKey(jobName);
        JobKey jobKey = JobKey.jobKey(jobName);
        scheduler.pauseTrigger(triggerKey);
        scheduler.unscheduleJob(triggerKey);
        scheduler.deleteJob(jobKey);
    }
}

ScheduledUtils 



import com.myapp.base.sys.job.entity.Scheduled;
import org.springframework.stereotype.Component;

import java.util.Collections;
import java.util.Hashtable;
import java.util.Map;
import java.util.concurrent.locks.ReentrantLock;

/**
 * 定时任务
 */
@Component
public class ScheduledUtils {


    private ReentrantLock lock = new ReentrantLock();

    private Map<String,Object> scheduledMap = Collections.synchronizedMap(new Hashtable<String,Object>());

    /**
     * 添加定时任务
     * @param scheduled
     * @return
     */
    public boolean addSchedulerJob(Scheduled scheduled){
        lock.lock();
        boolean flag =true;
        try{
            if(scheduledMap.containsKey(scheduled.getJobKey())){
                System.out.println("任务已添加");
                return flag;
            }
            QuartzUtils.createScheduleJob(scheduled);
            scheduledMap.put(scheduled.getJobKey(),"1");
        }catch (Exception e){
            System.out.println(e);
            flag = false;
        } finally {
            lock.unlock();
        }
        return flag;
    }

    /**
     * 更新定时任务
     * @param scheduled
     * @return
     */
    public boolean updateSchedulerJob(Scheduled scheduled){
        boolean flag =true;
        try{
            if(!scheduledMap.containsKey(scheduled.getJobKey())){
                addSchedulerJob(scheduled);
            }else{
                QuartzUtils.updateScheduleJob(scheduled);
            }
        }catch (Exception e){
            flag = false;
        }
        return flag;
    }

    public boolean deleteSchedulerJob(Scheduled scheduled){
        boolean flag =true;
        try{
            if(scheduledMap.containsKey(scheduled.getJobKey())){
                QuartzUtils.deleteScheduleJob(scheduled.getJobKey());
                scheduledMap.remove(scheduled.getJobKey());
            }
        }catch (Exception e){
            flag = false;
        }
        return flag;
    }


    public boolean runOnceSchedulerJob(Scheduled scheduled){
        boolean flag =true;
        try{
            if(scheduledMap.containsKey(scheduled.getJobKey())){
                QuartzUtils.runOnce(scheduled.getJobKey());
            }
        }catch (Exception e){
            flag = false;
        }
        return flag;
    }

}

service接口



import com.myapp.base.sys.job.entity.Scheduled;

import java.util.List;

public interface IScheduledService {

     
    boolean updateScheduled(Scheduled scheduled);

    boolean updateScheduledStartOrStop(Scheduled scheduled);

    boolean addScheduled(Scheduled scheduled);

    List<Scheduled> selectAllScheduled();

    boolean runOnce(Scheduled scheduled);
}

对应的实现类

这里实现类中实现了ApplicationRunner 中的run方法,作用在于,项目启动后,查询数据库中启用的定时工作,并逐一启用。

代码里 SysHelper.YES  等价与 “1” 表示的是任务的状态,启用与禁用。



import com.myapp.base.config.jobconfig.ScheduledUtils;
import com.myapp.base.sys.SysHelper;
import com.myapp.base.sys.job.entity.Scheduled;
import com.myapp.base.sys.job.mapper.ScheduledMapper;
import com.myapp.base.sys.job.service.IScheduledService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.ApplicationArguments;
import org.springframework.boot.ApplicationRunner;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;

import java.util.Date;
import java.util.List;

@Transactional
@Service
public class ScheduledService implements IScheduledService, ApplicationRunner {

    @Autowired
    ScheduledUtils scheduler;
    @Autowired
    ScheduledMapper scheduledMapper;


    /**
     * 查询数据库中可执行的
     * @param args
     * @throws Exception
     */
    @Override
    public void run(ApplicationArguments args) throws Exception {
        List<Scheduled> scheduleds = scheduledMapper.selectCanRun();
        if(scheduleds != null && scheduleds.size()>0){
            for(Scheduled item:scheduleds){
                scheduler.addSchedulerJob(item);
            }
        }


    }

    @Override
    public boolean updateScheduled(Scheduled scheduled) {
        int i = scheduledMapper.updateById(scheduled);
        if(i>0 && SysHelper.YES.equals(scheduled.getStatus())){
            return scheduler.updateSchedulerJob(scheduled);
        }else if(i>0){
            return scheduler.deleteSchedulerJob(scheduled);
        }

        return true;
    }


    @Override
    public boolean updateScheduledStartOrStop(Scheduled scheduled) {
        int i = scheduledMapper.updateById(scheduled);
        if(i>0){
            if(SysHelper.YES.equals(scheduled.getStatus())){
                return scheduler.addSchedulerJob(scheduled);
            }else{
                return scheduler.deleteSchedulerJob(scheduled);
            }
        }
        return false;
    }

    @Override
    public boolean addScheduled(Scheduled scheduled) {
        scheduled.setCreateTime(new Date());
        if(scheduledMapper.insert(scheduled)>0){
            return  scheduler.addSchedulerJob(scheduled);
        }
        return false;
    }

    @Override
    public List<Scheduled> selectAllScheduled() {
        return scheduledMapper.selectList(null);
    }

    @Override
    public boolean runOnce(Scheduled scheduled) {
        return scheduler.runOnceSchedulerJob(scheduled);
    }


}

定时任务的实现

创建类并继承QuartzJobBean类,实现其中的executeInternal方法。


import org.quartz.JobExecutionContext;
import org.quartz.JobExecutionException;
import org.springframework.scheduling.quartz.QuartzJobBean;

public class testJob extends QuartzJobBean {

    @Override
    protected void executeInternal(JobExecutionContext context) throws JobExecutionException {
        System.out.println("这里写你的定时任务");
    }


}

具体操作

前端页面,用vue简单搭了个界面,丑陋了点。

将我们创建的定时任务添加上去。

 注意:

1.类名一定要正确,因为我们是通过类名反射后获取对应的运行时类的,若类名不正确,就无法获取到类。

2.关键字不能与之前重复,因为我们是用map集合进行记录当前已运行的任务,关键字作为map的key若是重复了会导致该任务不会启动(启动时是先判断map中是否有该任务的,不想这样的,可自己改一下添加定时任务的逻辑)

具体controller和前端页面就不多说了,就是简单的传数据。

ScheduledUtils类

addSchedulerJob  添加任务

updateSchedulerJob 更新定时任务,主要是更新执行周期

deleteSchedulerJob 设置禁用时调用

runOnceSchedulerJob 需要立即执行时调用

水平有限,仅供参考。

  • 1
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
要特性 - 使用最新技术栈,社区资源丰富,基于Java 21(Core Module Support 17-21)、Spring Boot 3.2。 (Support Virtual Threads/fibre/loom) - 基于注解的动态查询(Specification),可根据需要扩充查询注解。 - 支持接口级别的功能权限,动态权限控制 - 支持数据字典,可方便地对一些状态进行管理 - 高效率开发,代码生成器可一键生成前后端代码 - 对一些常用前端组件封装:表格数据请求、数据字典等 - 前后端统一异常拦截处理,统一输出异常,避免繁琐的判断 - 使用ShardingSphere实现多数据源和读写分离。该方式针对MySQL数据库。对系统侵入性小。(只需引入依赖,并在yaml中配置数据源信息即可) [unicorn-starter](https://github.com/lWoHvYe/unicorn-starter)。 - 整合Redisson拓展Redis的功能,读写分离 - 整合消息队列RabbitMQ,实现消息通知、延迟消息,服务解耦。 - 各模块独立,基本可插拔:若只需查询注解等基础功能,只需引入Core模块即可,Beans, Security, Logging, 3rd Tools, Code Gen 模块可插拔, 除了传统To B业务,还可用于To C业务(see [OAuth2.0 part](unicorn-oauth2) ) #### 系统功能 - 用户管理:提供用户的相关配置,新增用户后,默认密码为123456 - 角色管理:对权限与菜单进行分配,菜单权限、数据权限(Draft)、接口权限(_In Progress_) - 菜单管理:已实现菜单动态路由,后端可配置化,支持多级菜单 - 部门管理:可配置系统组织架构,树形表格展示(Draft) - 岗位管理:配置各个部门的职位(Draft) - 字典管理:可维护常用一些固定的数据,如:状态,性别等 - 系统日志:记录用户操作日志与异常日志,方便开发人员定位排错 - 定时任务:整合Quartz定时任务,加入任务日志,任务运行情况一目了然 - 代码生成:高灵活度生成前后端代码,减少大量重复的工作任务(逆向有很多方案,这种基于template的有一定的灵活性) - 邮件工具:配合富文本,发送html格式的邮件 #### 项目结构 项目采用按功能分模块的开发方式,结构如下 - `unicorn-core` 系统的Core模块,BaseClass及各种Util,(基于Multi-Release JAR Files,Support Java 17 - 21) - `unicorn-beans` 基础Beans的Definition及Configuration,To C业务可只引入该dependency - `unicorn-sys-api` Sys Module基础实体及API,方便服务拆分 - `unicorn-security` 系统权限模块,包含权限配置管理等。 - `unicorn-logging` 系统的日志模块,其他模块如果需要记录日志需要引入该模块,亦可自行实现 - `unicorn-tp-tools-kotlin` 第三方工具模块,包含:邮件、S3,可视情况引入 - `unicorn-code-gen-kotlin` 系统的代码生成模块。这部分待优化,亦非必须模块 - `unicorn-starter` [启动类(Maven),项目入口,包含模块及组件配置(DB读写分离 + Cache读写分离)](https://github.com/lWoHvYe/unicorn-starter) - `valentine-starter` 启动配置示例(Gradle),尝试Kotlin/Kotlinx - `unicorn-oauth2` OAuth2 Sample,AuthorizationServer, OAuth2Client + Gateway, ResourceServer #### 详细结构 ``` - unicorn-core 公共模块 ## 项目备注 1、该资源内项目代码都经过测试运行成功,功能ok的情况下才上传的,请放心下载使用! 2、本项目适合计算机相关专业(如计科、人工智能、通信工程、自动化、电子信息等)的在校学生、老师或者企业员工下载学习,也适合小白学习进阶,当然也可作为毕设项目、课程设计、作业、项目初期立项演示等。 3、如果基础还行,也可在此代码基础上进行修改,以实现其他功能,也可用于毕设、课设、作业等。 下载后请首先打开README.md文件(如有),仅供学习参考, 切勿用于商业用途。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值