当前位置:首页 > 科技  > 软件

告别硬编码,SpringBoot实现动态增删启停定时任务

来源: 责编: 时间:2023-10-31 16:46:26 429观看
导读在spring boot项目中,可以通过@EnableScheduling注解和@Scheduled注解实现定时任务,也可以通过SchedulingConfigurer接口来实现定时任务。但是这两种方式不能动态添加、删除、启动、停止任务。要实现动态增删启停定时任

在spring boot项目中,可以通过@EnableScheduling注解和@Scheduled注解实现定时任务,也可以通过SchedulingConfigurer接口来实现定时任务。但是这两种方式不能动态添加、删除、启动、停止任务。pvS28资讯网——每日最新资讯28at.com

要实现动态增删启停定时任务功能,比较广泛的做法是集成Quartz框架。但是本人的开发原则是:在满足项目需求的情况下,尽量少的依赖其它框架,避免项目过于臃肿和复杂。pvS28资讯网——每日最新资讯28at.com

查看spring-context这个jar包中org.springframework.scheduling.ScheduledTaskRegistrar这个类的源代码,发现可以通过改造这个类就能实现动态增删启停定时任务功能。pvS28资讯网——每日最新资讯28at.com

定时任务列表页定时任务列表页pvS28资讯网——每日最新资讯28at.com

定时任务执行日志定时任务执行日志pvS28资讯网——每日最新资讯28at.com

添加执行定时任务的线程池配置类pvS28资讯网——每日最新资讯28at.com

@Configuration  public class SchedulingConfig {      @Bean      public TaskScheduler taskScheduler() {          ThreadPoolTaskScheduler taskScheduler = new ThreadPoolTaskScheduler();          // 定时任务执行线程池核心线程数          taskScheduler.setPoolSize(4);          taskScheduler.setRemoveOnCancelPolicy(true);          taskScheduler.setThreadNamePrefix("TaskSchedulerThreadPool-");          return taskScheduler;      }  }

添加ScheduledFuture的包装类。ScheduledFuture是ScheduledExecutorService定时任务线程池的执行结果。pvS28资讯网——每日最新资讯28at.com

public final class ScheduledTask {        volatile ScheduledFuture<?> future;        /**       * 取消定时任务       */      public void cancel() {          ScheduledFuture<?> future = this.future;          if (future != null) {              future.cancel(true);          }      }  }

添加Runnable接口实现类,被定时任务线程池调用,用来执行指定bean里面的方法。pvS28资讯网——每日最新资讯28at.com

public class SchedulingRunnable implements Runnable {        private static final Logger logger = LoggerFactory.getLogger(SchedulingRunnable.class);        private String beanName;        private String methodName;        private String params;        public SchedulingRunnable(String beanName, String methodName) {          this(beanName, methodName, null);      }        public SchedulingRunnable(String beanName, String methodName, String params) {          this.beanName = beanName;          this.methodName = methodName;          this.params = params;      }        @Override      public void run() {          logger.info("定时任务开始执行 - bean:{},方法:{},参数:{}", beanName, methodName, params);          long startTime = System.currentTimeMillis();            try {              Object target = SpringContextUtils.getBean(beanName);                Method method = null;              if (StringUtils.isNotEmpty(params)) {                  method = target.getClass().getDeclaredMethod(methodName, String.class);              } else {                  method = target.getClass().getDeclaredMethod(methodName);              }                ReflectionUtils.makeAccessible(method);              if (StringUtils.isNotEmpty(params)) {                  method.invoke(target, params);              } else {                  method.invoke(target);              }          } catch (Exception ex) {              logger.error(String.format("定时任务执行异常 - bean:%s,方法:%s,参数:%s ", beanName, methodName, params), ex);          }            long times = System.currentTimeMillis() - startTime;          logger.info("定时任务执行结束 - bean:{},方法:{},参数:{},耗时:{} 毫秒", beanName, methodName, params, times);      }        @Override      public boolean equals(Object o) {          if (this == o) return true;          if (o == null || getClass() != o.getClass()) return false;          SchedulingRunnable that = (SchedulingRunnable) o;          if (params == null) {              return beanName.equals(that.beanName) &&                      methodName.equals(that.methodName) &&                      that.params == null;          }            return beanName.equals(that.beanName) &&                  methodName.equals(that.methodName) &&                  params.equals(that.params);      }        @Override      public int hashCode() {          if (params == null) {              return Objects.hash(beanName, methodName);          }            return Objects.hash(beanName, methodName, params);      }  }

添加定时任务注册类,用来增加、删除定时任务。pvS28资讯网——每日最新资讯28at.com

@Component  public class CronTaskRegistrar implements DisposableBean {        private final Map<Runnable, ScheduledTask> scheduledTasks = new ConcurrentHashMap<>(16);        @Autowired      private TaskScheduler taskScheduler;        public TaskScheduler getScheduler() {          return this.taskScheduler;      }        public void addCronTask(Runnable task, String cronExpression) {          addCronTask(new CronTask(task, cronExpression));      }        public void addCronTask(CronTask cronTask) {          if (cronTask != null) {              Runnable task = cronTask.getRunnable();              if (this.scheduledTasks.containsKey(task)) {                  removeCronTask(task);              }                this.scheduledTasks.put(task, scheduleCronTask(cronTask));          }      }        public void removeCronTask(Runnable task) {          ScheduledTask scheduledTask = this.scheduledTasks.remove(task);          if (scheduledTask != null)              scheduledTask.cancel();      }        public ScheduledTask scheduleCronTask(CronTask cronTask) {          ScheduledTask scheduledTask = new ScheduledTask();          scheduledTask.future = this.taskScheduler.schedule(cronTask.getRunnable(), cronTask.getTrigger());            return scheduledTask;      }          @Override      public void destroy() {          for (ScheduledTask task : this.scheduledTasks.values()) {              task.cancel();          }            this.scheduledTasks.clear();      }  }

添加定时任务示例类pvS28资讯网——每日最新资讯28at.com

@Component("demoTask")  public class DemoTask {      public void taskWithParams(String params) {          System.out.println("执行有参示例任务:" + params);      }        public void taskNoParams() {          System.out.println("执行无参示例任务");      }  }

定时任务数据库表设计pvS28资讯网——每日最新资讯28at.com

定时任务数据库表设计定时任务数据库表设计pvS28资讯网——每日最新资讯28at.com

public class SysJobPO {      /**       * 任务ID       */      private Integer jobId;      /**       * bean名称       */      private String beanName;      /**       * 方法名称       */      private String methodName;      /**       * 方法参数       */      private String methodParams;      /**       * cron表达式       */      private String cronExpression;      /**       * 状态(1正常 0暂停)       */      private Integer jobStatus;      /**       * 备注       */      private String remark;      /**       * 创建时间       */      private Date createTime;      /**       * 更新时间       */      private Date updateTime;        public Integer getJobId() {          return jobId;      }        public void setJobId(Integer jobId) {          this.jobId = jobId;      }        public String getBeanName() {          return beanName;      }        public void setBeanName(String beanName) {          this.beanName = beanName;      }        public String getMethodName() {          return methodName;      }        public void setMethodName(String methodName) {          this.methodName = methodName;      }        public String getMethodParams() {          return methodParams;      }        public void setMethodParams(String methodParams) {          this.methodParams = methodParams;      }        public String getCronExpression() {          return cronExpression;      }        public void setCronExpression(String cronExpression) {          this.cronExpression = cronExpression;      }        public Integer getJobStatus() {          return jobStatus;      }        public void setJobStatus(Integer jobStatus) {          this.jobStatus = jobStatus;      }        public String getRemark() {          return remark;      }        public void setRemark(String remark) {          this.remark = remark;      }        public Date getCreateTime() {          return createTime;      }        public void setCreateTime(Date createTime) {          this.createTime = createTime;      }        public Date getUpdateTime() {          return updateTime;      }        public void setUpdateTime(Date updateTime) {          this.updateTime = updateTime;      }    }

新增定时任务pvS28资讯网——每日最新资讯28at.com

新增定时任务新增定时任务pvS28资讯网——每日最新资讯28at.com

boolean success = sysJobRepository.addSysJob(sysJob);  if (!success)      return OperationResUtils.fail("新增失败");  else {      if (sysJob.getJobStatus().equals(SysJobStatus.NORMAL.ordinal())) {          SchedulingRunnable task = new SchedulingRunnable(sysJob.getBeanName(), sysJob.getMethodName(), sysJob.getMethodParams());          cronTaskRegistrar.addCronTask(task, sysJob.getCronExpression());      }  }    return OperationResUtils.success();

修改定时任务,先移除原来的任务,再启动新任务pvS28资讯网——每日最新资讯28at.com

boolean success = sysJobRepository.editSysJob(sysJob);  if (!success)      return OperationResUtils.fail("编辑失败");  else {      //先移除再添加      if (existedSysJob.getJobStatus().equals(SysJobStatus.NORMAL.ordinal())) {          SchedulingRunnable task = new SchedulingRunnable(existedSysJob.getBeanName(), existedSysJob.getMethodName(), existedSysJob.getMethodParams());          cronTaskRegistrar.removeCronTask(task);      }        if (sysJob.getJobStatus().equals(SysJobStatus.NORMAL.ordinal())) {          SchedulingRunnable task = new SchedulingRunnable(sysJob.getBeanName(), sysJob.getMethodName(), sysJob.getMethodParams());          cronTaskRegistrar.addCronTask(task, sysJob.getCronExpression());      }  }    return OperationResUtils.success();

删除定时任务pvS28资讯网——每日最新资讯28at.com

boolean success = sysJobRepository.deleteSysJobById(req.getJobId());  if (!success)      return OperationResUtils.fail("删除失败");  else{      if (existedSysJob.getJobStatus().equals(SysJobStatus.NORMAL.ordinal())) {          SchedulingRunnable task = new SchedulingRunnable(existedSysJob.getBeanName(), existedSysJob.getMethodName(), existedSysJob.getMethodParams());          cronTaskRegistrar.removeCronTask(task);      }  }    return OperationResUtils.success();

定时任务启动/停止状态切换pvS28资讯网——每日最新资讯28at.com

if (existedSysJob.getJobStatus().equals(SysJobStatus.NORMAL.ordinal())) {      SchedulingRunnable task = new SchedulingRunnable(existedSysJob.getBeanName(), existedSysJob.getMethodName(), existedSysJob.getMethodParams());      cronTaskRegistrar.addCronTask(task, existedSysJob.getCronExpression());  } else {      SchedulingRunnable task = new SchedulingRunnable(existedSysJob.getBeanName(), existedSysJob.getMethodName(), existedSysJob.getMethodParams());      cronTaskRegistrar.removeCronTask(task);  }

添加实现了CommandLineRunner接口的SysJobRunner类,当spring boot项目启动完成后,加载数据库里状态为正常的定时任务。另外,关注公众号码猿技术专栏,回复关键词9527,送你一份Spring Cloud Alibaba实战视频教程!pvS28资讯网——每日最新资讯28at.com

@Service  public class SysJobRunner implements CommandLineRunner {        private static final Logger logger = LoggerFactory.getLogger(SysJobRunner.class);        @Autowired      private ISysJobRepository sysJobRepository;        @Autowired      private CronTaskRegistrar cronTaskRegistrar;        @Override      public void run(String... args) {          // 初始加载数据库里状态为正常的定时任务          List<SysJobPO> jobList = sysJobRepository.getSysJobListByStatus(SysJobStatus.NORMAL.ordinal());          if (CollectionUtils.isNotEmpty(jobList)) {              for (SysJobPO job : jobList) {                  SchedulingRunnable task = new SchedulingRunnable(job.getBeanName(), job.getMethodName(), job.getMethodParams());                  cronTaskRegistrar.addCronTask(task, job.getCronExpression());              }                logger.info("定时任务已加载完毕...");          }      }  }

工具类SpringContextUtils,用来从spring容器里获取beanpvS28资讯网——每日最新资讯28at.com

@Component  public class SpringContextUtils implements ApplicationContextAware {        private static ApplicationContext applicationContext;        @Override      public void setApplicationContext(ApplicationContext applicationContext)              throws BeansException {          SpringContextUtils.applicationContext = applicationContext;      }        public static Object getBean(String name) {          return applicationContext.getBean(name);      }        public static <T> T getBean(Class<T> requiredType) {          return applicationContext.getBean(requiredType);      }        public static <T> T getBean(String name, Class<T> requiredType) {          return applicationContext.getBean(name, requiredType);      }        public static boolean containsBean(String name) {          return applicationContext.containsBean(name);      }        public static boolean isSingleton(String name) {          return applicationContext.isSingleton(name);      }        public static Class<? extends Object> getType(String name) {          return applicationContext.getType(name);      }  }

本文完,参考本文代码可成功运行,亲测!pvS28资讯网——每日最新资讯28at.com

本文链接:http://www.28at.com/showinfo-26-16141-0.html告别硬编码,SpringBoot实现动态增删启停定时任务

声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。邮件:2376512515@qq.com

上一篇: 我们一起聊聊 Rust 变量,你学会了吗?

下一篇: .Net虚拟机(CLR/JIT)加密原理(版权保护)

标签:
  • 热门焦点
  • 俄罗斯:将审查iPhone等外国公司设备 保数据安全

    iPhone和特斯拉都属于在各自领域领头羊的品牌,推出的产品也也都是数一数二的,但对于一些国家而言,它们的产品可靠性和安全性还是在限制范围内。近日,俄罗斯联邦通信、信息技术
  • JavaScript 混淆及反混淆代码工具

    介绍在我们开始学习反混淆之前,我们首先要了解一下代码混淆。如果不了解代码是如何混淆的,我们可能无法成功对代码进行反混淆,尤其是使用自定义混淆器对其进行混淆时。什么是混
  • 一年经验在二线城市面试后端的经验分享

    忠告这篇文章只适合2年内工作经验、甚至没有工作经验的朋友阅读。如果你是2年以上工作经验,请果断划走,对你没啥帮助~主人公这篇文章内容来自 「升职加薪」星球星友 的投稿,坐
  • 企业采用CRM系统的11个好处

    客户关系管理(CRM)软件可以为企业提供很多的好处,从客户保留到提高生产力。  CRM软件用于企业收集客户互动,以改善客户体验和满意度。  CRM软件市场规模如今超过580
  • 雅柏威士忌多款单品价格大跌,泥煤顶流也不香了?

    来源 | 烈酒商业观察编 | 肖海林今年以来,威士忌市场开始出现了降温迹象,越来越多不断暴涨的网红威士忌也开始悄然回归市场理性。近日,LVMH集团旗下苏格兰威士忌品牌雅柏(Ardbeg
  • 猿辅导与新东方的两种“归途”

    作者|卓心月 出品|零态LT(ID:LingTai_LT)如何成为一家伟大企业?答案一定是对&ldquo;势&rdquo;的把握,这其中最关键的当属对企业战略的制定,且能够站在未来看现在,即使这其中的
  • 自研Exynos回归!三星Galaxy S24系列将提供Exynos和骁龙双版本

    年初,全新的三星Galaxy S23系列发布,包含Galaxy S23、Galaxy S23+和Galaxy S23 Ultra三个版本,全系搭载超频版骁龙8 Gen 2,虽同样采用台积电4nm工艺制
  • 首发天玑9200+ iQOO Neo8系列发布首销售价2299元起

    2023年5月23日晚,iQOO Neo8系列正式发布。其中,Neo系列首款Pro之作——iQOO Neo8 Pro强悍登场,限时售价3099元起;价位段最强性能手机iQOO Neo8同期上市
  • 联想YOGA 16s 2022笔记本将要推出,屏幕支持触控功能

    联想此前宣布,将于11月2日19:30召开联想秋季轻薄新品发布会,推出联想 YOGA 16s 2022 笔记本等新品。官方称,YOGA 16s 2022 笔记本将搭载 16 英寸屏幕,并且是一
Top