SpringBoot中并发定时任务的实现、动态定时任务的实现(看这一篇就够了)推荐

一、在JAVA开发领域,目前可以通过以下几种方式进行定时任务

1、单机部署模式

Timer:jdk中自带的一个定时调度类,可以简单的实现按某一频度进行任务执行。提供的功能比较单一,无法实现复杂的调度任务。
ScheduledExecutorService:也是jdk自带的一个基于线程池设计的定时任务类。其每个调度任务都会分配到线程池中的一个线程执行,所以其任务是并发执行的,互不影响。
Spring Task:Spring提供的一个任务调度工具,支持注解和配置文件形式,支持Cron表达式,使用简单但功能强大。
Quartz:一款功能强大的任务调度器,可以实现较为复杂的调度功能,如每月一号执行、每天凌晨执行、每周五执行等等,还支持分布式调度,就是配置稍显复杂。

2、分布式集群模式(不多介绍,简单提一下)

问题:

  1. I、如何解决定时任务的多次执行?
  2. II、如何解决任务的单点问题,实现任务的故障转移?

问题I的简单思考:

  1. 1、固定执行定时任务的机器(可以有效避免多次执行的情况 ,缺点就是单点故障问题)。
  2. 2、借助Redis的过期机制和分布式锁。
  3. 3、借助mysql的锁机制等。

成熟的解决方案:

1、Quartz:可以去看看这篇文章[Quartz分布式]( https://www.jb51.net/article/102869.htm)。
2、elastic-job:(https://github.com/elasticjob/elastic-job-lite)当当开发的弹性分布式任务调度系统,采用zookeeper实现分布式协调,实现任务高可用以及分片。
3、xxl-job:(https://github.com/xuxueli/xxl-job)是大众点评员发布的分布式任务调度平台,是一个轻量级分布式任务调度框架。
4、saturn:(https://github.com/vipshop/Saturn) 是唯品会提供一个分布式、容错和高可用的作业调度服务框架。

二、SpringTask实现定时任务(这里是基于springboot)

1、简单的定时任务实现

使用方式:

  1. 使用@EnableScheduling注解开启对定时任务的支持。
  2. 使用@Scheduled 注解即可,基于corn、fixedRate、fixedDelay等一些定时策略来实现定时任务。

使用缺点:

  1. 1、多个定时任务使用的是同一个调度线程,所以任务是阻塞执行的,执行效率不高。
  2. 2、其次如果出现任务阻塞,导致一些场景的定时计算没有实际意义,比如每天12点的一个计算任务被阻塞到1点去执行,会导致结果并非我们想要的。

使用优点:

  1. 1、配置简单
  2. 2、适用于单个后台线程执行周期任务,并且保证顺序一致执行的场景

 源码分析:

//默认使用的调度器
if(this.taskScheduler == null) {
 this.localExecutor = Executors.newSingleThreadScheduledExecutor();
 this.taskScheduler = new ConcurrentTaskScheduler(this.localExecutor);
}
//可以看到SingleThreadScheduledExecutor指定的核心线程为1,说白了就是单线程执行
public static ScheduledExecutorService newSingleThreadScheduledExecutor() {
 return new DelegatedScheduledExecutorService
 (new ScheduledThreadPoolExecutor(1));
}
//利用了DelayedWorkQueue延时队列作为任务的存放队列,这样便可以实现任务延迟执行或者定时执行
public ScheduledThreadPoolExecutor(int corePoolSize) {
 super(corePoolSize, Integer.MAX_VALUE, 0, NANOSECONDS,
  new DelayedWorkQueue());
}

2、实现并发的定时任务

使用方式:

方式一:由1中我们知道之所以定时任务是阻塞执行,是配置的线程池决定的,那就好办了,换一个不就行了!直接上代码:

@Configuration
 public class ScheduledConfig implements SchedulingConfigurer {

 @Autowired
 private TaskScheduler myThreadPoolTaskScheduler;

 @Override
 public void configureTasks(ScheduledTaskRegistrar scheduledTaskRegistrar) {
  //简单粗暴的方式直接指定
  //scheduledTaskRegistrar.setScheduler(Executors.newScheduledThreadPool(5));
  //也可以自定义的线程池,方便线程的使用与维护,这里不多说了
  scheduledTaskRegistrar.setTaskScheduler(myThreadPoolTaskScheduler);
 }
 }

 @Bean(name = "myThreadPoolTaskScheduler")
 public TaskScheduler getMyThreadPoolTaskScheduler() {
 ThreadPoolTaskScheduler taskScheduler = new ThreadPoolTaskScheduler();
 taskScheduler.setPoolSize(10);
 taskScheduler.setThreadNamePrefix("Haina-Scheduled-");
 taskScheduler.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
 //调度器shutdown被调用时等待当前被调度的任务完成
 taskScheduler.setWaitForTasksToCompleteOnShutdown(true);
 //等待时长
 taskScheduler.setAwaitTerminationSeconds(60);
 return taskScheduler;
 } 

方式二:方式一的本质改变了任务调度器默认使用的线程池,接下来这种是不改变调度器的默认线程池,而是把当前任务交给一个异步线程池去执行

首先使用@EnableAsync 启用异步任务
然后在定时任务的方法加上@Async即可,默认使用的线程池为SimpleAsyncTaskExecutor(该线程池默认来一个任务创建一个线程,就会不断创建大量线程,极有可能压爆服务器内存。当然它有自己的限流机制,这里就不多说了,有兴趣的自己翻翻源码~)
项目中为了更好的控制线程的使用,我们可以自定义我们自己的线程池,使用方式@Async("myThreadPool")

废话太多,直接上代码:

 @Scheduled(fixedRate = 1000*10,initialDelay = 1000*20)
 @Async("myThreadPoolTaskExecutor")
 //@Async
 public void scheduledTest02(){
  System.out.println(Thread.currentThread().getName()+"--->xxxxx--->"+Thread.currentThread().getId());
 }

 //自定义线程池
 @Bean(name = "myThreadPoolTaskExecutor")
 public TaskExecutor getMyThreadPoolTaskExecutor() {
  ThreadPoolTaskExecutor taskExecutor = new ThreadPoolTaskExecutor();
  taskExecutor.setCorePoolSize(20);
  taskExecutor.setMaxPoolSize(200);
  taskExecutor.setQueueCapacity(25);
  taskExecutor.setKeepAliveSeconds(200);
  taskExecutor.setThreadNamePrefix("Haina-ThreadPool-");
  // 线程池对拒绝任务(无线程可用)的处理策略,目前只支持AbortPolicy、CallerRunsPolicy;默认为后者
  taskExecutor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
  //调度器shutdown被调用时等待当前被调度的任务完成
  taskExecutor.setWaitForTasksToCompleteOnShutdown(true);
  //等待时长
  taskExecutor.setAwaitTerminationSeconds(60);
  taskExecutor.initialize();
  return taskExecutor;
 }

线程池的使用心得(后续有专门文章来探讨)

java中提供了ThreadPoolExecutor和ScheduledThreadPoolExecutor,对应与spring中的ThreadPoolTaskExecutor和ThreadPoolTaskScheduler,但是在原有的基础上增加了新的特性,在spring环境下更容易使用和控制。
使用自定义的线程池能够避免一些默认线程池造成的内存溢出、阻塞等等问题,更贴合自己的服务特性
使用自定义的线程池便于对项目中线程的管理、维护以及监控。
即便在非spring环境下也不要使用java默认提供的那几种线程池,坑很多,阿里代码规约不说了吗,得相信大厂!!!

三、动态定时任务的实现

问题:
使用@Scheduled注解来完成设置定时任务,但是有时候我们往往需要对周期性的时间的设置会做一些改变,或者要动态的启停一个定时任务,那么这个时候使用此注解就不太方便了,原因在于这个注解中配置的cron表达式必须是常量,那么当我们修改定时参数的时候,就需要停止服务,重新部署。

解决办法:

方式一:实现SchedulingConfigurer接口,重写configureTasks方法,重新制定Trigger,核心方法就是addTriggerTask(Runnable task, Trigger trigger) ,不过需要注意的是,此种方式修改了配置值后,需要在下一次调度结束后,才会更新调度器,并不会在修改配置值时实时更新,实时更新需要在修改配置值时额外增加相关逻辑处理。

@Configuration
 public class ScheduledConfig implements SchedulingConfigurer {

 @Autowired
 private TaskScheduler myThreadPoolTaskScheduler;

 @Override
 public void configureTasks(ScheduledTaskRegistrar scheduledTaskRegistrar) {
  //scheduledTaskRegistrar.setScheduler(Executors.newScheduledThreadPool(5));
  scheduledTaskRegistrar.setTaskScheduler(myThreadPoolTaskScheduler);
  //可以实现动态调整定时任务的执行频率
  scheduledTaskRegistrar.addTriggerTask(
    //1.添加任务内容(Runnable)
    () -> System.out.println("cccccccccccccccc--->" + Thread.currentThread().getId()),
    //2.设置执行周期(Trigger)
    triggerContext -> {
     //2.1 从数据库动态获取执行周期
     String cron = "0/2 * * * * ? ";
     //2.2 合法性校验.
 //     if (StringUtils.isEmpty(cron)) {
 //      // Omitted Code ..
 //     }
      //2.3 返回执行周期(Date)
      return new CronTrigger(cron).nextExecutionTime(triggerContext);
     }
   );
 }
 }

方式二:使用threadPoolTaskScheduler类可实现动态添加删除功能,当然也可实现执行频率的调整

首先,我们要认识下这个调度类,它其实是对java中ScheduledThreadPoolExecutor的一个封装改进后的产物,主要改进有以下几点:

  1. 1、提供默认配置,因为是ScheduledThreadPoolExecutor,所以只有poolSize这一个默认参数。
  2. 2、支持自定义任务,通过传入Trigger参数。
  3. 3、对任务出错处理进行优化,如果是重复性的任务,不抛出异常,通过日志记录下来,不影响下次运行,如果是只执行一次的任务,将异常往上抛。

顺便说下ThreadPoolTaskExecutor相对于ThreadPoolExecutor的改进点:

  1. 1、提供默认配置,原生的ThreadPoolExecutor的除了ThreadFactory和RejectedExecutionHandler其他没有默认配置
  2. 2、实现AsyncListenableTaskExecutor接口,支持对FutureTask添加success和fail的回调,任务成功或失败的时候回执行对应回调方法。
  3. 3、因为是spring的工具类,所以抛出的RejectedExecutionException也会被转换为spring框架的TaskRejectedException异常(这个无所谓)
  4. 4、提供默认ThreadFactory实现,直接通过参数重载配置

扯了这么多,还是直接上代码:

 @Component
 public class DynamicTimedTask {

  private static final Logger logger = LoggerFactory.getLogger(DynamicTimedTask.class);

  //利用创建好的调度类统一管理
  //@Autowired
  //@Qualifier("myThreadPoolTaskScheduler")
  //private ThreadPoolTaskScheduler myThreadPoolTaskScheduler;

  //接受任务的返回结果
  private ScheduledFuture<?> future;

  @Autowired
  private ThreadPoolTaskScheduler threadPoolTaskScheduler;

  //实例化一个线程池任务调度类,可以使用自定义的ThreadPoolTaskScheduler
  @Bean
  public ThreadPoolTaskScheduler threadPoolTaskScheduler() {
   ThreadPoolTaskScheduler executor = new ThreadPoolTaskScheduler();
   return new ThreadPoolTaskScheduler();
  }

  /**
  * 启动定时任务
  * @return
  */
  public boolean startCron() {
   boolean flag = false;
   //从数据库动态获取执行周期
   String cron = "0/2 * * * * ? ";
   future = threadPoolTaskScheduler.schedule(new CheckModelFile(),cron);
   if (future!=null){
    flag = true;
    logger.info("定时check训练模型文件,任务启动成功!!!");
   }else {
    logger.info("定时check训练模型文件,任务启动失败!!!");
   }
   return flag;
  }

  /**
  * 停止定时任务
  * @return
  */
  public boolean stopCron() {
   boolean flag = false;
   if (future != null) {
    boolean cancel = future.cancel(true);
    if (cancel){
     flag = true;
     logger.info("定时check训练模型文件,任务停止成功!!!");
    }else {
     logger.info("定时check训练模型文件,任务停止失败!!!");
    }
   }else {
    flag = true;
    logger.info("定时check训练模型文件,任务已经停止!!!");
   }
   return flag;
  }

  class CheckModelFile implements Runnable{

   @Override
   public void run() {
    //编写你自己的业务逻辑
    System.out.print("模型文件检查完毕!!!")
   }
  }

 }

四、总结

到此基于springtask下的定时任务的简单使用算是差不多了,其中不免有些错误的地方,或者理解有偏颇的地方欢迎大家提出来!
基于分布式集群下的定时任务使用,后续有时间再继续!!!

以上所述是小编给大家介绍的SpringBoot并发定时任务动态定时任务实现详解整合,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对我们网站的支持!

(0)

相关推荐

  • SpringBoot 并发登录人数控制的实现方法

    通常系统都会限制同一个账号的登录人数,多人登录要么限制后者登录,要么踢出前者,Spring Security 提供了这样的功能,本文讲解一下在没有使用Security的时候如何手动实现这个功能 demo 技术选型 SpringBoot JWT Filter Redis + Redisson JWT(token)存储在Redis中,类似 JSessionId-Session的关系,用户登录后每次请求在Header中携带jwt 如果你是使用session的话,也完全可以借鉴本文的思路,只是代码上需要

  • spring boot高并发下耗时操作的实现方法

    高并发下的耗时操作 高并发下,就是请求在一个时间点比较多时,很多写的请求打过来时,你的服务器承受很大的压力,当你的一个请求处理时间长时,这些请求将会把你的服务器线程耗尽,即你的主线程池里的线程将不会再有空闲状态的,再打过来的请求,将会是502了. 请求流程图 http1 http2 http3 thread1 thread2 thread3 解决方案 使用DeferredResult来实现异步的操作,当一个请求打过来时,先把它放到一个队列时,然后在后台有一个订阅者,有相关主题的消息发过来时,这个

  • spring-boot 多线程并发定时任务的解决方案

    刚刚看了下Spring Boot实现定时任务的文章,感觉还不错.Spring Boot 使用Spring自带的Schedule来实现定时任务变得非常简单和方便.在这里个大家分享下. 开启缓存注解 @SpringBootApplication @EnableScheduling //开启定时任务 public class Application { public static void main(String[] args) { SpringApplication.run(Application.

  • SpringBoot后端接口的实现(看这一篇就够了)

    摘要:本文演示如何构建起一个优秀的后端接口体系,体系构建好了自然就有了规范,同时再构建新的后端接口也会十分轻松. 一个后端接口大致分为四个部分组成:接口地址(url).接口请求方式(get.post等).请求数据(request).响应数据(response).如何构建这几个部分每个公司要求都不同,没有什么"一定是最好的"标准,但一个优秀的后端接口和一个糟糕的后端接口对比起来差异还是蛮大的,其中最重要的关键点就是看是否规范! 本文就一步一步演示如何构建起一个优秀的后端接口体系,体系构建

  • Java中你真的会用Constructor构造器吗之看完本篇你就真的会了

    引言 相信大家对于java里的构造器应该都是有了解的,这次我们来了解一些构造器的不同使用方式,了解构造器的调用顺序,最后可以灵活的在各种情况下定义使用构造器,进一步优化我们的代码: 构造器简介 还是简单介绍一下构造器到底是什么吧, 构造器是类中一种特殊的方法,通过调用构造器来完成对象的创建,以及对象属性的初始化操作. 构造器定义方式: [修饰符列表] 构造器名(形式参数列表){ 构造方法体; } 构造器有以下几个特点: 构造器名和类名一致: 构造器用来创建对象,以及完成属性初始化操作: 构造器返

  • SpringBoot中并发定时任务的实现、动态定时任务的实现(看这一篇就够了)推荐

    一.在JAVA开发领域,目前可以通过以下几种方式进行定时任务 1.单机部署模式 Timer:jdk中自带的一个定时调度类,可以简单的实现按某一频度进行任务执行.提供的功能比较单一,无法实现复杂的调度任务. ScheduledExecutorService:也是jdk自带的一个基于线程池设计的定时任务类.其每个调度任务都会分配到线程池中的一个线程执行,所以其任务是并发执行的,互不影响. Spring Task:Spring提供的一个任务调度工具,支持注解和配置文件形式,支持Cron表达式,使用简单

  • 浅谈SpringBoot集成Quartz动态定时任务

    SpringBoot自带schedule 沿用的springboot少xml配置的优良传统,本身支持表达式等多种定时任务 注意在程序启动的时候加上@EnableScheduling @Scheduled(cron="0/5 * * * * ?") public void job(){ System.out.println("每五秒执行一次"); } 为什么要使用Quartz 多任务情况下,quartz更容易管理,可以实现动态配置 执行时间表达式: 表达式示例: 集成

  • SpringBoot实现动态定时任务

    项目情况: 在当前项目中需要一个定时任务来清除过期的校验码,如果使用数据库存储过程的话不方便维护.因此采用SpringBoot自带的方式来设置定时任务. 技术说明: SpringBoot自带的方式有两种可以实现: 一种是使用@Scheduled注解的方式,只需要在启动类或者它所在的类上添加@EnableScheduling注解允许执行定时任务,并且设置Schecduled注解的参数,诸如: 1.cron是设置定时执行的表达式,如 0 0/5 * * * ?每隔五分钟执行一次 2.zone表示执行

  • SpringBoot开发实战系列之动态定时任务

    目录 前言 代码编写 效果演示 启动 修改 停止 后记 前言 定时器是我们项目中经常会用到的,SpringBoot使用@Scheduled注解可以快速启用一个简单的定时器(详情请看我们之前的博客<SpringBoot系列--定时器>),然而这种方式的定时器缺乏灵活性,如果需要对定时器进行调整,需要重启项目才生效,本文记录SpringBoot如何灵活配置动态定时任务 代码编写 首先先建表,重要字段:唯一表id.Runnable任务类.Cron表达式,其他的都是一些额外补充字段 DROP TABL

  • SpringBoot设置动态定时任务的方法详解

    之前写过文章记录怎么在SpringBoot项目中简单使用定时任务,不过由于要借助cron表达式且都提前定义好放在配置文件里,不能在项目运行中动态修改任务执行时间,实在不太灵活. 经过网上搜索学习后,特此记录如何在SpringBoot项目中实现动态定时任务. 因为只是一个demo,所以只引入了需要的依赖: <dependencies> <dependency> <groupId>org.springframework.boot</groupId> <ar

  • SpringBoot+Quartz实现动态定时任务

    本文实例为大家分享了springBoot+Quartz实现动态定时任务的具体代码,供大家参考,具体内容如下 目前常用的几种任务调度 Timer,简单无门槛,一般也没人用. spring @Scheduled注解,一般集成于项目中,小任务很方便. 开源工具 Quartz,分布式集群开源工具,以下两个分布式任务应该都是基于Quartz实现的,可以说是中小型公司必选,当然也视自身需求而定. 分布式任务 XXL-JOB,是一个轻量级分布式任务调度框架,支持通过 Web 页面对任务进行 CRUD 操作,支

  • Springboot实现动态定时任务流程详解

    目录 一.静态 二.动态 1.基本代码 2.方案详解 2.1 初始化 2.2 单次执行 2.3 停止任务 2.4 启用任务 三.小结 一.静态 静态的定时任务可以直接使用注解@Scheduled,并在启动类上配置@EnableScheduling即可 @PostMapping("/list/test1") @Async @Scheduled(cron = "0 * * * * ?") public void test1() throws Exception { Ob

随机推荐