zoukankan      html  css  js  c++  java
  • SpringBoot整合定时任务和异步任务处理

     

     

    SpringBoot定时任务schedule讲解
    简介:讲解什么是定时任务和常见定时任务区别

    一.定时任务

    1、常见定时任务 Java自带的java.util.Timer类
    timer:配置比较麻烦,时间延后问题,不推荐
    timertask:不推荐

    2、Quartz框架(复杂定时任务可以使用,spring 或springmv项目)
    配置更简单
    xml或者注解

    具体说明后续......

    3、SpringBoot使用注解方式开启定时任务(springboot项目推荐使用)
    1)启动类里面 @EnableScheduling开启定时任务,自动扫描
    2)定时任务业务类 加注解 @Component被容器扫描
    3)定时执行的方法加上注解 @Scheduled(fixedRate=2000) 定期执行一次 2秒

     

    4、SpringBoot常用定时任务表达式配置和在线生成器   crontab 工具 https://tool.lu/crontab/

    1)cron 定时任务表达式 @Scheduled(cron="*/1 * * * * *") 表示每秒

     

    cron表达式,有专门的语法,而且感觉有点绕人,不过简单来说,大家记住一些常用的用法即可,特殊的语法可以单独去查。
    cron一共有7位,但是最后一位是年,可以留空,所以我们可以写6位:
    * 第一位,表示秒,取值0-59
    * 第二位,表示分,取值0-59
    * 第三位,表示小时,取值0-23
    * 第四位,日期天/日,取值1-31
    * 第五位,日期月份,取值1-12
    * 第六位,星期,取值1-7,星期一,星期二...,注:不是第1周,第二周的意思
              另外:1表示星期天,2表示星期一。
    * 第7为,年份,可以留空,取值1970-2099

     


    cron中,还有一些特殊的符号,含义如下:
    (*)星号:可以理解为每的意思,每秒,每分,每天,每月,每年...
    (?)问号:问号只能出现在日期和星期这两个位置,表示这个位置的值不确定,每天3点执行,所以第六位星期的位置,我们是不需要关注的,就是不确定的值。同时:日期和星期是两个相互排斥的元素,通过问号来表明不指定值。比如,1月10日,比如是星期1,如果在星期的位置是另指定星期二,就前后冲突矛盾了。
    (-)减号:表达一个范围,如在小时字段中使用“10-12”,则表示从10到12点,即10,11,12
    (,)逗号:表达一个列表值,如在星期字段中使用“1,2,4”,则表示星期一,星期二,星期四
    (/)斜杠:如:x/y,x是开始值,y是步长,比如在第一位(秒) 0/15就是,从0秒开始,每15秒,最后就是0,15,30,45,60    另:*/y,等同于0/y


    下面列举几个例子:
    0 0 3 * * ?     每天3点执行
    0 5 3 * * ?     每天3点5分执行
    0 5 3 ? * *     每天3点5分执行,与上面作用相同
    0 5/10 3 * * ?  每天3点的 5分,15分,25分,35分,45分,55分这几个时间点执行
    0 10 3 ? * 1    每周星期天,3点10分 执行,注:1表示星期天    
    0 10 3 ? * 1#3  每个月的第三个星期,星期天 执行,#号只能出现在星期的位置

     

     


    2)fixedRate: 定时多久执行一次(上一次开始执行时间点后xx秒再次执行;)
    3)fixedDelay: 上一次执行结束时间点后xx秒再次执行
    4)fixedDelayString: 字符串形式,可以通过配置文件指定

    5.定时任务单线程/多线程问题处理

    1)问题描述,sprinboot按照如上方式配置定时任务,如果项目中有不止一个定时任务,并且有部分定时任务(执行时间较长或监听类定时任务),会导致后面其他的定时任务无法执行

    2)问题处理:提供两种方案

    方案一:

    出现此问题的原因,是springboot默认的定时任务为单线程执行,即项目中的多个定时任务在同一线程中串行执行,一旦其中的某个定时任务出现执行时间较长或监听服务、死循环等情况,其他定时任务将一直处于等待,

    无法执行

    附springboot源码如下:

    if (this.taskScheduler == null) {
        this.localExecutor = Executors.newSingleThreadScheduledExecutor();//此处导致单线程问题
        this.taskScheduler = new ConcurrentTaskScheduler(this.localExecutor);
    }

    解决方案:创建配置文件 ScheduleConfig.java

    package com.liansheng.authority_service.cronJob.config;

    import org.springframework.context.annotation.Configuration;
    import org.springframework.scheduling.annotation.SchedulingConfigurer;
    import org.springframework.scheduling.config.ScheduledTaskRegistrar;

    import java.util.concurrent.Executors;

    /**
    * 多线程执行定时任务
    * @author fjt
    * 2020年3月27日
    */
    //@Configuration
    //所有的定时任务都放在一个线程池中,定时任务启动时使用不同都线程。
    public class ScheduleConfig implements SchedulingConfigurer {
    @Override
    public void configureTasks(ScheduledTaskRegistrar taskRegistrar) {
    //重新自定义定时任务线程池
    taskRegistrar.setScheduler(Executors.newScheduledThreadPool(15));//考虑具体项目中定时任务的多少及执行效率
    }
    }

    此种方案下,如果某个定时任务出现处理时间较长或卡死、死循环之类的问题,此定时任务会一直等待,后期不会再次执行,不会影响线程的后期使用及其他定时任务。


    方案二:
      通过springboot的异步任务处理
    ··第一步:配置一个线程池,可以不配置,用默认线程池
    第二步:在你的定时任务方法或类 上增加 @Async (如果使用自己配置的线程池,应为@Async(“自己创建的线程池名称”),如果没有配置线程池,则@Async)
    第三步:启动项目,发现各定时任务全部启动,互不影响

    此种方案下,如果某个定时任务出现处理时间较长或卡死、死循环之类的问题,此定时任务会定时在指定时间利用新的线程再次启动该定时任务,会消耗掉大量线程,如果此任务一直无法释放的话,运行时间较长后会导致系统奔溃,
    如果使用此种方案建议检查定时任务代码质量及执行效率。优点是一个定时任务在出现异常后,不会影响下次指定时间再次执行。

     以上问题是在项目中遇到的问题及处理方案总结,各有优缺点,需要根据项目具体情况,选择使用,也可以混合使用。

    二、SpringBoot2.x异步任务实战(核心知识)
    简介:讲解什么是异步任务,和使用SpringBoot2.x开发异步任务实战
    1、什么是异步任务和使用场景:适用于处理log、发送邮件、短信……等  

    一个service方法内需要异步调用多个方法,这些方法可以异步执行的情况下使用,提高后台效率。
    下单接口->查库存 100
    余额校验 150
    风控用户100
    ....


    2、启动类里面使用@EnableAsync注解开启功能,自动扫描

    3、定义异步任务类并使用@Component标记组件被容器扫描,异步方法加上@Async
    注意点:
    1)要把异步任务封装到类里面,不能直接写到Controller
    2)增加Future<String> 返回结果 AsyncResult<String>("task执行完成");
    3)如果需要拿到结果 需要判断全部的 task.isDone()
    4、通过注入方式,注入到controller里面,如果测试前后区别则改为同步则把Async注释掉

  • 相关阅读:
    vue 下载模板
    vue 使用XLSX 导入表格
    el-select 同时传递多个参数 id value.
    关于前端node 内存溢出
    js判断输入是否含有空格
    python中的内置函数总结
    Python的数据类型和常用方法大全
    简单认识python的数据类型和语法
    Part1.1 、RabbitMQ 操作使用
    Part1.2 、RabbitMQ -- Publish/Subscribe 【发布和订阅】
  • 原文地址:https://www.cnblogs.com/fujingtao5470/p/11659849.html
Copyright © 2011-2022 走看看