【ThreadPoolTaskScheduler实现动态定时任务添加、删除、修改】

本文介绍了如何在Springboot项目中使用ThreadPoolTaskScheduler实现动态添加、删除和修改定时任务,通过Cron表达式控制任务执行频率,包括BaskTask接口的定义和ScheduledTaskManager类的实现。
摘要由CSDN通过智能技术生成

总体设计

在Springboot项目中使用ThreadPoolTaskScheduler添加定时任务,结合Cron表达式来实现

注册 ThreadPoolTaskScheduler

在配置文件中添加如下代码,并开启@EnableAsync 注解启动异步线程池

    @Bean
    @ConditionalOnMissingBean(ThreadPoolTaskScheduler.class)
    public ThreadPoolTaskScheduler threadPoolTaskScheduler() {
        ThreadPoolTaskScheduler threadPoolTaskScheduler = new ThreadPoolTaskScheduler();
        threadPoolTaskScheduler.setPoolSize(5);                        // 线程池大小,根据实际需要添加
        threadPoolTaskScheduler.setThreadNamePrefix("taskScheduler-");   // 线程名称
        threadPoolTaskScheduler.setAwaitTerminationSeconds(600);         // 等待时长
        // 调度器shutdown被调用时等待当前被调度的任务完成
        threadPoolTaskScheduler.setWaitForTasksToCompleteOnShutdown(false); 
        return threadPoolTaskScheduler;
    }

创建具体任务的执行对象 BaskTask

/**
 * 基础任务接口
 *
 * @author ryenlii
 */
public interface BaskTask extends Runnable {

    /**
     * 获取执行频率
     */
    String getCron();

    /**
     * 获取执行对象名称
     */
    String getName();

    /**
     * 执行任务逻辑
     */
    void execute();
}

创建定时任务处理中心 ScheduledTaskManager


import XXXX.ScheduledTask;
import XXXX.BaskTask;
import lombok.extern.slf4j.Slf4j;
import org.springframework.scheduling.concurrent.ThreadPoolTaskScheduler;
import org.springframework.stereotype.Component;

import javax.annotation.Resource;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;

/**
 * 动态添加任务配置
 *
 * @author ryenlii
 */
@Slf4j
@Component
public class ScheduledTaskManager {

    /**
     * 以下都是线程安全的集合类。
     */
    private final Map<String, ScheduledTask> scheduledTasks = new ConcurrentHashMap<>();

    @Resource
    private ThreadPoolTaskScheduler syncScheduler;

    /**
     * 添加一个动态任务。
     *
     * @param task 要添加的任务对象。
     * @return 添加成功返回true,否则返回false。
     */
    public boolean add(BaskTask task) {
        // 记录任务总数。
        log.info("任务总数: {}", scheduledTasks.keySet());

        String taskName = task.getName();
        // 优化查找性能
        if (scheduledTasks.containsKey(taskName)) {
            updateCron(taskName, task.getCron());
        } else {
            // 新增
            addTask(taskName, task);
        }
        return true;
    }


    /**
     * 添加一个新的定时任务
     *
     * @param name 任务名称
     * @param task 定时任务(Runnable对象)
     */
    public void addTask(String name, BaskTask task) {
        if (scheduledTasks.containsKey(name)) {
            throw new IllegalArgumentException("A task with the same name already exists: " + name);
        }

        ScheduledTask scheduledTask = new ScheduledTask(syncScheduler, task);
        scheduledTask.schedule();
        scheduledTasks.put(name, scheduledTask);

        log.info("定时任务{}已成功创建,Cron表达式:{}", name, task.getCron());
    }

    /**
     * 更新已安排任务的Cron表达式
     *
     * @param name              任务名称
     * @param newCronExpression 新的Cron表达式
     */
    public void updateCron(String name, String newCronExpression) {
        ScheduledTask scheduledTask = scheduledTasks.get(name);
        if (scheduledTask == null) {
            throw new IllegalArgumentException("No scheduled task found with the given name: " + name);
        }

        scheduledTask.updateCronExpression(newCronExpression);

        log.info("定时任务{}的Cron表达式已成功更新为:{}", name, newCronExpression);
    }

    /**
     * 停止并移除一个定时任务
     *
     * @param name 任务名称
     */
    public void stopTask(String name) {
        ScheduledTask scheduledTask = scheduledTasks.remove(name);
        if (scheduledTask != null) {
            scheduledTask.cancel();
            log.info("定时任务{}已成功停止", name);
        } else {
            log.warn("尝试停止的任务{}不存在", name);
        }
    }

}

创建任务处理ScheduledTask


import XXX.BaskTask;
import cn.hutool.core.date.DateUtil;
import cn.hutool.log.Log;
import cn.hutool.log.LogFactory;
import org.springframework.scheduling.TaskScheduler;
import org.springframework.scheduling.support.CronTrigger;

import java.util.Date;
import java.util.concurrent.ScheduledFuture;

/**
 * 定时任务处理对象
 *
 * @author ryenlii
 */
public class ScheduledTask {
    private final Log log = LogFactory.get();
    private final BaskTask task;
    private String cronExpression;
    private ScheduledFuture<?> scheduledFuture;
    private final TaskScheduler taskScheduler;

    public ScheduledTask(TaskScheduler taskScheduler, BaskTask task) {
        this.taskScheduler = taskScheduler;
        this.task = task;
        this.cronExpression = task.getCron();
    }

    public void schedule() {
        this.scheduledFuture = taskScheduler.schedule(task, scheduledConfig -> {
            Date nextExecutionTime = new CronTrigger(cronExpression).nextExecutionTime(scheduledConfig);
            log.info("{}执行定时任务【{}】的时间:{}", task.getName(), cronExpression, DateUtil.formatDateTime(nextExecutionTime));
            return nextExecutionTime;
        });
    }

    public void updateCronExpression(String newCronExpression) {
        cancel();
        this.cronExpression = newCronExpression;
        schedule();
    }

    public void cancel() {
        if (this.scheduledFuture != null) {
            this.scheduledFuture.cancel(false);
        }
    }

}

项目启动后执行定时任务

此处针对某些单独执行的任务,进行手动添加,也可以通过获取Spring的对象,批量添加


import XXX.BaskTask;
import XXX.DataCrawlerTask;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.ApplicationArguments;
import org.springframework.boot.ApplicationRunner;
import org.springframework.core.annotation.Order;
import org.springframework.scheduling.annotation.EnableScheduling;
import org.springframework.stereotype.Component;


@Order(-1)
@Component
@SuppressWarnings("all")
@Slf4j
@EnableScheduling
public class ScheduleRunner implements ApplicationRunner {

    @Autowired
    private ScheduledTaskManager task;

    @Override
    public void run(ApplicationArguments args) throws Exception {
        // 爬取标准信息
        BaskTask baskTask = new DataCrawlerTask();
        task.add(baskTask);
    }

}

批量添加

import ……

public class ScheduleRunner implements ApplicationRunner {
    @Autowired
    private ApplicationContext applicationContext;

    @Autowired
    private ScheduledTaskManager task;

    @Override
    public void run(ApplicationArguments args) throws Exception {
        // 获取所有的定时任务
        Map<String, BaskTask> map = applicationContext.getBeansOfType(BaskTask.class);

        // 遍历注册
        for (String key : map.keySet()) {
            // baskTask:回调内部的run方法
            BaskTask baskTask = map.get(key);
            // 立即执行
            // baskTask.execute();
            task.add(baskTask);
        }
    }
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

今宵酒千樽

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值