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注释掉