SpringBoot定时工作和异步操作
日常求赞,感激老板。欢送关注公众号:其实是白羊。干货继续更新中......
一、定时工作
在做业务时总会有这样的场景:在特定工夫去执行某些逻辑。这其实就是定时工作的利用场景,比方:须要每月一日给用户发上月数据总结等场景。
1.技术
实现定时工作的技术很多
- Timer:JDK自带的java.util.Timer其实更相似于定时器,可实现提早执行和依照肯定频率执行,也能够指定某个工夫执行,应用较少
- ScheduledExecutorService:也是JDK自带的,是基于线程池设计的定时工作类,依据Executors创立时的线程数量去执行具体任务(多个线程数量就是每个人物调配一个线程)
- Spring Task:即明天要介绍了配角,是Spring自带的,当然这里通过Springboot应用。
- Quartz:开源的调度框架,性能更加弱小
2.注解应用
1)@EnableScheduling
开启定时工作,可标注在启动类或者任何配置类上(能扫描到对象上都能够)
2)@Scheduled
配置具体的工作执行规定,可标注在能被扫描的类的办法上,属性有:
- fixedRate:上一个工作开始工夫和下一个工作开始工夫的工夫距离(毫秒)
- fixedDelay:上一个工作的完结工夫和下一个工作的开始工夫的工夫距离(毫秒)
- initialDelay:第一次执行提早执行的工夫(毫秒)
- cron:通过cron表达式来配置执行机会
3.cron表达式
定时工作的场景:提早执行、肯定频率执行、指定工夫执行
这里的cron表达式即为了形容工作执行的工夫规定。cron由6-7个元素组成,他们之间应用空格来宰割,以此代表:
- 秒:0~59
- 分:0~59
- 时:0~23
- 日:1~31(具体月的最初一天也可能是30)
- 月:1~12
- 星期:1~7(留神1为周日)
- 年:1970~2099
除了下面的数字元素值还有上面几个非凡的元素值:
- *:示意任意一个值都会触发,七个元素地位中都能够呈现,如在分钟呈现即示意每分钟都回去执行
- ?:和*相似,但只会呈现在日和星期(这两个地位只能有一个是具备意义的形容,如:要么是每个月的3号要么是每个月的每个星期的周三),当其中一个配置了有意义的值,另一个写?
- -:示意范畴(至),七个元素地位中都能够呈现,如分钟里呈现1-7则示意1分钟到7分钟每分钟执行一次
- /:示意从开始工夫每隔多长时间执行一次,七个元素地位中都能够呈现,如分钟里呈现1/7则示意1分触发一次1+7=8分触发一次
- ,:示意枚举值,七个元素地位中都能够呈现,如分钟里呈现1,7则示意1分和7分各执行一次
- L:示意最初,只会呈现在日和星期地位,日示意最初一天,星期则会搭配数字如5L示意最初一个周四
- W:示意无效工作日(周一到周五),只会呈现在日期里,后面会搭配数字,如5W:如果5号是周六那么执行工夫为周五即4号;如果5号是周日那么执行工夫为下周一即6号;(即零碎会举荐离明天最近的一个工作日,但留神不会进行逾越查找:如果是31W且31号是周日,那么会在29号执行)
- LW:连用示意最初一个工作日,只会呈现在日期里
- “#”:(前后要加数字)示意第几个星期几,只会呈现在星期里,如4#2 即示意第二个周三(后面示意周几,前面示意第几个)
理解了下面的表达式规定就能够写出满足条件的cron表达式了,倡议多设计几个场景练习下
4.单线程和多线程执行
如果就按下面的配置写好工作A(fixedRate=2000)和工作B(fixedRate=3000)间接启动执行的话,会存在以下问题:
- 如果A1工作执行工夫超过2s那么原定于A1工作开始时2s后执行的工作A2就不能按时执行
- 工作A和工作B要交替执行
下面的问题都不能让咱们的工作依照各自规定的执行打算去执行,归根到底还是所有的工作都在一个线程里进行,所以做不到异步的并发执行,那这是就能够通过多线程来实现各个工作之间异步的并发执行
这里咱们能够应用Spring的@Async注解来实现异步
二、异步
SpringBoot同样反对@Async来实现异步(减少了主动配置可间接应用)
1.注解
1)@EnableAsync
标注在启动类或配置类上,表是开启异步
2)@Async
可标注在类上(能被spring容器扫描到的类上)或办法上
标注在类上则这个类里的办法都被示意为异步
2.异步办法两种返回值
- 不须要返回值:void
须要返回值:
@Async public Future<String> doTaskOne() throws Exception { return new AsyncResult<>("工作一实现"); }
3.主动配置
应用@Async来实现异步,其底层还是应用了多线程来进行实现的,那么这个多线程或者线程池是在哪里设置或配置的呢?咱们都直到SpringBoot退出了大量的主动配置,咱们在spring-boot-autoconfigure包上面能够找到task包下的TaskExecutionAutoConfiguration类中:
@Lazy@Bean(name = { APPLICATION_TASK_EXECUTOR_BEAN_NAME, AsyncAnnotationBeanPostProcessor.DEFAULT_TASK_EXECUTOR_BEAN_NAME })@ConditionalOnMissingBean(Executor.class)//这个注解的意思:当在容器中没有发现Executor这个类则会加载这个bean,能够了解为此处为默认缺省对象//返回的是一个spring为咱们提供的线程池 public ThreadPoolTaskExecutor applicationTaskExecutor(TaskExecutorBuilder builder) { return builder.build(); }
找到TaskExecutorBuilder:
@Bean @ConditionalOnMissingBean public TaskExecutorBuilder taskExecutorBuilder() { TaskExecutionProperties.Pool pool = this.properties.getPool(); TaskExecutorBuilder builder = new TaskExecutorBuilder(); builder = builder.queueCapacity(pool.getQueueCapacity()); builder = builder.corePoolSize(pool.getCoreSize()); builder = builder.maxPoolSize(pool.getMaxSize()); builder = builder.allowCoreThreadTimeOut(pool.isAllowCoreThreadTimeout()); builder = builder.keepAlive(pool.getKeepAlive()); builder = builder.threadNamePrefix(this.properties.getThreadNamePrefix()); builder = builder.customizers(this.taskExecutorCustomizers.orderedStream()::iterator); builder = builder.taskDecorator(this.taskDecorator.getIfUnique()); return builder; }
这里的properties是在TaskExecutionProperties加载进来的,默认的参数:
private int queueCapacity = Integer.MAX_VALUE;private int coreSize = 8; private int maxSize = Integer.MAX_VALUE;private boolean allowCoreThreadTimeout = true;
依据上面可知,也能够从yml/properties文件中自定义配置
@ConfigurationProperties("spring.task.execution")
除了下面的办法咱们还能够定义本人的Bean来替换默认缺省Bean,依据@ConditionalOnMissingBean(Executor.class)可知,咱们只须要在配置类里加上:
@Bean public Executor taskExecutor() { ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor(); executor.setCorePoolSize(外围线程数量); executor.setMaxPoolSize(最大线程数量); executor.setQueueCapacity(工作队列大小); executor.initialize(); return executor; }
三、最初
依据下面的介绍,要实现异步的定时工作就能够开展,工作办法下面加上@Async来实现
点个赞啊亲
如果你认为本文对你有帮忙,能够「在看/转发/赞/star」,多谢如果你还发现了更好或不同的想法,还能够在留言区一起探讨下
欢送关注公众号:「其实是白羊」干货继续更新中......