spring schedule实现动态配置执行时间

作者:g-Jack 时间:2022-09-06 18:47:30 

spring schedule 动态配置执行时间

之前saas平台实现动态修改定时任务的时间,都是通过xx-job这样的框架来实现,这样我们可以单独一个服务来管理我们整个saas平台的定时任务,但是最近给银行做的一个小项目,需要本地化部署,所以我不想弄很多的服务,并且他们并没有要求修改以后即时生效,所以我直接采用了 spring schedule结合mysql动态配置执行时间。

之前我们用的schedule通过注解的方式,只能用静态的corn表达式,如果想实现动态的需要实现SchedulingConfigurer,并且通过注解@EnableScheduling。如下:


package com.zqf.marketing.task;  
import com.zqf.db.marketingrobot.sys.model.RobotSysSwitch;
import com.zqf.marketing.sys.service.SwitchService;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Lazy;
import org.springframework.scheduling.Trigger;
import org.springframework.scheduling.TriggerContext;
import org.springframework.scheduling.annotation.EnableScheduling;
import org.springframework.scheduling.annotation.SchedulingConfigurer;
import org.springframework.scheduling.config.ScheduledTaskRegistrar;
import org.springframework.scheduling.support.CronTrigger;
import org.springframework.stereotype.Service;
import java.util.Date;

/**
* @author zhenghao
* @description
* @date 2019/1/22 21:50
*/

@Lazy(false)
@Service
@EnableScheduling
public class TestTaskService implements SchedulingConfigurer {
   private static Logger log = LoggerFactory.getLogger(TestTaskService.class);
   @Autowired
   private SwitchService switchService;

private String SpringDynamicCronTask() {
       String cron = "0/5 * * * * ?";
       //从数据库获得配置的corn表达式
       RobotSysSwitch switchById = switchService.getSwitchById(5L);
       cron = switchById.getSwitchFlag();
       log.info(cron);
       return cron;
   }

@Override
   public void configureTasks(ScheduledTaskRegistrar scheduledTaskRegistrar) {
       scheduledTaskRegistrar.addTriggerTask(new Runnable() {
           @Override
           public void run() {
               // 任务逻辑
               log.info("task_task_tak");
           }
       }, new Trigger() {
           @Override
           public Date nextExecutionTime(TriggerContext triggerContext) {
               String s = SpringDynamicCronTask();
               // 任务触发,可修改任务的执行周期
               CronTrigger trigger = new CronTrigger(s);
               Date nextExec = trigger.nextExecutionTime(triggerContext);
               return nextExec;
           }
       });  
   }
}

这样我们就可以动态的修改task的执行时间,生效时间为,上一个任务的执行周期,也可以满足我们现在需求,这样就可以实习项目更加的灵活!

@schedule注解动态配置时间间隔

动态配置时间间隔是通过自己实现的任务注册到任务调度实现的,并在每次调度的时候更改下次调度时间间隔,如果任务阻塞或者挂掉了就不会再被调度了,如果设置时间过长,到下次调度就需要等待很长时间。


import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.scheduling.Trigger;
import org.springframework.scheduling.TriggerContext;
import org.springframework.scheduling.annotation.EnableScheduling;
import org.springframework.scheduling.annotation.SchedulingConfigurer;
import org.springframework.scheduling.config.ScheduledTaskRegistrar;
import org.springframework.scheduling.support.PeriodicTrigger;
import org.springframework.stereotype.Component;
import java.util.Date;

@Component
@EnableScheduling
public class DynamicScheduleTaskSecond implements SchedulingConfigurer {
   private static final long WEEK_MILLIS = 604800000;
   private static final long MIN_MILLIS = 1000;
   private static long period = 1000;
   static long l = System.currentTimeMillis();
   @Autowired
   SetPeriod setPeriod;

public static long getPeriod() {
       return period;
   }

public static void setPeriod(long period) {
       DynamicScheduleTaskSecond.period = period;
   }

@Override
   public void configureTasks(ScheduledTaskRegistrar taskRegistrar) {
       taskRegistrar.addTriggerTask(new Runnable() {
           @Override
           public void run() {
               try {
                   setPeriod.update(period);
                   System.out.println("abc");
                   Long last = System.currentTimeMillis() - l;
                   l = System.currentTimeMillis();
                   System.out.println(last);
               } catch (Exception e) {
                   e.printStackTrace();
               }
           }
       }, new Trigger() {
           @Override
           public Date nextExecutionTime(TriggerContext triggerContext) {
               if (period < MIN_MILLIS || period > WEEK_MILLIS)
                   period = MIN_MILLIS;
               PeriodicTrigger periodicTrigger = new PeriodicTrigger(period);
               Date nextExecDate = periodicTrigger.nextExecutionTime(triggerContext);
               return nextExecDate;
           }
       });
   }
}

import org.springframework.stereotype.Component;
@Component
public class SetPeriod {
   private static Long maxPeriod = 1000l;
   public void update(Long period) {
       maxPeriod += 1000;
       setScheduleConfig(maxPeriod);
   }

public boolean setScheduleConfig(Long period) {
       DynamicScheduleTaskSecond.setPeriod(period);
       return true;
   }
}

上面是实现动态调度的一个简单实例,下面说一下基本原理。

动态调度功能主要是实现SchedulingConfigurer函数式接口,接口中的方法configureTasks的参数是重点。


@FunctionalInterface
public interface SchedulingConfigurer {
/**
 * Callback allowing a {@link org.springframework.scheduling.TaskScheduler
 * TaskScheduler} and specific {@link org.springframework.scheduling.config.Task Task}
 * instances to be registered against the given the {@link ScheduledTaskRegistrar}.
 * @param taskRegistrar the registrar to be configured.
 */
void configureTasks(ScheduledTaskRegistrar taskRegistrar);
}

看名字 ScheduledTaskRegistrar就知道是一个调度任务注册类,调用这个类的addTriggerTask方法需要两个参数


   public void addTriggerTask(Runnable task, Trigger trigger) {
       this.addTriggerTask(new TriggerTask(task, trigger));
   }

一个是任务线程这个最后说,先说一下第二个Trigger,这是一个设置任务触发时间的接口,具体的实现有两个类,一个是CronTrigger对应的就是cron类型的时间设置,一个是PeriodicTrigger对应的就是FixDelay和FixRate两种方式的时间设置,实例中使用的是后者。


public interface Trigger {
   @Nullable
   Date nextExecutionTime(TriggerContext var1);
}

接口方法参数是一个TriggerContext,这个参数就是任务触发的上下文,里面保存着上一次任务开始时间和结束时间和实际执行用时,自己需要实现这个nextExecutionTime方法根据上一次任务执行时间来返回一个新的Date时间,new一个新的periodicTrigger对象初始化period时间间隔为新的时间间隔用nextExecutionTime方法就可以了根据上下文时间返回一个新的任务调度时间了,但是period的时间不能太长也不能太短最好设置一个区间,这样可以避免很多粗心的错误导致的麻烦,到此完美解决动态设置任务调度时间间隔功能。

再说一下第一个线程任务中的需要做的事,执行的任务需要在其他的具体类中实现,然后在这个线程中调用,然后每次在调度任务的时候就要根据时间业务重新设置时间间隔,比如读配置后改变时间间隔,也就是调度和具体的任务形成一个环,调度执行具体的任务后,具体的任务在设置调度的时间间隔。

来源:https://blog.csdn.net/hao134838/article/details/86604722

标签:spring,schedule,执行时间
0
投稿

猜你喜欢

  • Android仿qq分组管理的第三方库

    2023-08-27 05:31:21
  • java.math.BigDecimal的用法及加减乘除计算

    2022-01-15 15:55:20
  • Android 使用 okhttp3和retrofit2 进行单文件和多文件上传

    2023-04-29 07:33:22
  • Java8如何使用Lambda表达式简化代码详解

    2023-02-20 04:44:58
  • Android 开发之BottomBar+ViewPager+Fragment实现炫酷的底部导航效果

    2022-01-07 19:39:20
  • 浅谈java中Math.random()与java.util.random()的区别

    2023-11-26 16:37:16
  • C#利用性能计数器监控网络状态

    2022-01-05 00:13:53
  • SpringBoot整合Zookeeper详细教程

    2022-07-24 11:33:09
  • MybatisPlus使用Wrapper实现条件查询功能

    2021-11-29 10:21:08
  • Android编程中Tween动画和Frame动画实例分析

    2023-12-03 06:31:56
  • IOS与网页JS交互详解及实例

    2023-07-08 11:58:20
  • Android TabHost如何实现顶部选项卡

    2023-04-13 01:08:14
  • maven中配置项目的jdk版本无效的排查方式

    2023-07-18 21:43:42
  • C# WPF Image控件的绑定方法

    2023-09-03 06:41:42
  • java随机验证码生成实现实例代码

    2022-05-18 03:03:31
  • spring cloud gateway网关路由分配代码实例解析

    2021-06-09 02:54:08
  • Struts2 的国际化实现方式示例

    2022-08-10 07:26:45
  • Unity3D绘制地形的实现方法

    2022-12-01 01:06:46
  • java多线程-同步块实例讲解

    2022-06-21 02:10:41
  • android通过gps获取定位的位置数据和gps经纬度    

    2023-04-26 04:51:59
  • asp之家 软件编程 m.aspxhome.com