查看原文
其他

Spring Boot 如何解决多个定时任务阻塞问题?

终码一生 2022-09-22

The following article is from 码猿技术专栏 Author 不才陈某

点击“终码一生”,关注,置顶公众号

每日技术干货,第一时间送达!



今天这篇文章介绍一下Spring Boot 中如何开启多线程定时任务?



1

为什么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()



2

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


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


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.................");
}



3

总结


本篇文章介绍了 Spring Boot 中实现多线程定时任务的三种方案,你喜欢哪一种?


PS:防止找不到本篇文章,可以收藏点赞,方便翻阅查找哦。


往期推荐



从 Java 12 到 Java 17 那些激动人心的新特性

Logback日志这样配置,让你性能提升10倍!

每日开源 | 真香啊!基于SpringBoot 的CMS系统,拿去吧

卧槽!一文搞懂 Redis 挖矿原理及如何防范?

每日开源 | 一个 Java 接口快速开发框架:magic-api

彻底搞懂 MyBaits 中#{}和${}的区别,#{}如何防止注入?


您可能也对以下帖子感兴趣

文章有问题?点此查看未经处理的缓存