软件编程
位置:首页>> 软件编程>> java编程>> Spring Boot多个定时任务阻塞问题的解决方法

Spring Boot多个定时任务阻塞问题的解决方法

作者:不才陈某  发布时间:2023-09-20 11:43:36 

标签:springboot,定时,任务

前言

为什么Spring Boot 定时任务是单线程的?

想要解释为什么,一定要从源码入手,直接从@EnableScheduling这个注解入手,找到了这个ScheduledTaskRegistrar类,其中有一段代码如下:

protected void scheduleTasks() {
 if (this.taskScheduler == null) {
  this.localExecutor = Executors.newSingleThreadScheduledExecutor();
  this.taskScheduler = new ConcurrentTaskScheduler(this.localExecutor);
 }
}

如果taskScheduler为null,则创建单线程的线程池:Executors.newSingleThreadScheduledExecutor()。

多线程定时任务如何配置?

下面介绍三种方案配置多线程下的定时任务。

1、重写SchedulingConfigurer#configureTasks()

直接实现SchedulingConfigurer这个接口,设置taskScheduler,代码如下:

@Configuration
public class ScheduleConfig implements SchedulingConfigurer {
   @Override
   public void configureTasks(ScheduledTaskRegistrar taskRegistrar) {
       //设定一个长度10的定时任务线程池
       taskRegistrar.setScheduler(Executors.newScheduledThreadPool(10));
   }
}

2、通过配置开启

Spring Boot quartz 已经提供了一个配置用来配置线程池的大小,如下;

spring.task.scheduling.pool.size=10

只需要在配置文件中添加如上的配置即可生效!

3、结合@Async

@Async这个注解都用过,用来开启异步任务的,使用@Async这个注解之前一定是要先配置线程池的,配置如下:

@Bean
   public ThreadPoolTaskExecutor taskExecutor() {
       ThreadPoolTaskExecutor poolTaskExecutor = new ThreadPoolTaskExecutor();
       poolTaskExecutor.setCorePoolSize(4);
       poolTaskExecutor.setMaxPoolSize(6);
       // 设置线程活跃时间(秒)
       poolTaskExecutor.setKeepAliveSeconds(120);
       // 设置队列容量
       poolTaskExecutor.setQueueCapacity(40);
       poolTaskExecutor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
       // 等待所有任务结束后再关闭线程池
       poolTaskExecutor.setWaitForTasksToCompleteOnShutdown(true);
       return poolTaskExecutor;
   }

然后在@Scheduled方法上标注@Async这个注解即可实现多线程定时任务,代码如下:

@Async
   @Scheduled(cron = "0/2 * * * * ? ")
   public void test2() {
       System.out.println("..................执行test2.................");
   }

来源:https://mp.weixin.qq.com/s/b_pIs6nKiINRDzowmxW2Ww

0
投稿

猜你喜欢

手机版 软件编程 asp之家 www.aspxhome.com