当前位置: 首页 > news >正文

Spring @Scheduled单线程单实例的坑

文章目录

    • 前言
    • 背景
    • 验证
    • 解决方案

前言

在 Java Spring 项目中经常会用 @Scheduled 来实现一些定时任务的场景,有必要了解一些它使用时的问题和内部实现机制。本文是偶然间发现的一个问题,刷新了我的认知,分享给大家。

其他相关文章:Spring @Scheduled 多线程配置

背景

在 Spring Web 项目中,使用了多个 @Scheduled 来做任务的定时跑批,发现与预期的效果不一致

比如三个 @Scheduled 定时任务,在三个不同的类里面

  • 任务1(class1):每10秒钟执行一次
  • 任务2(class2):每10秒钟执行一次
  • 任务3(class3):每10秒钟执行一次

预期效果:

三个任务是三个独立的定时任务线程池在控制,每10秒钟执行一次,任务之间互不影响

实际效果:

项目启动之后,发现这三个任务是串行执行,第一个任务未执行完的情况下,其他两个任务也会等待。

验证

启动项目之后,分别在三个任务上打上断点

spring @Scheduled 注解的处理类源码:

  • org.springframework.scheduling.annotation.ScheduledAnnotationBeanPostProcessor
    • org.springframework.scheduling.config.ScheduledTaskRegistrar#scheduleTasks 初始化方法

在项目启动过程中只进来了一次,taskScheduler 实例化赋值一次,而且是单线程的线程池 SingleThreadScheduledExecutor

this.localExecutor = Executors.newSingleThreadScheduledExecutor();

请添加图片描述

观察 thread 名称,分别在每个 @Scheduled 方法中都打断点

第一个 FaceTask#start() 方法,threadName = pool-8-thread-1

请添加图片描述

第二个 ProfileTask#start() 方法,threadName = pool-8-thread-1

请添加图片描述

第三个 TmpFaceTask#faceDelayPush() 方法,threadName = pool-8-thread-1

请添加图片描述

通过源码+代码调试可以得到结论,在同一个 Spring 容器中(项目中),使用多个 @Scheduled 定时任务,每个标记有 @Scheduled 任务之间是串行执行的,使用的是同一个线程池,此线程池默认通过 Executors.newSingleThreadScheduledExecutor() 创建的单线程(核心线程=最大线程)的线程池。

解决方案

如何达到不同定时任务之间互不影响,都采用独立的定时任务线程池来执行呢?

由于 @Scheduled 的实现提供的扩展点比较单一,我们只能采用 API 编程的方式来完成了。

  • 每个类中来创建定时任务线程池
  • 调用需要执行的业务逻辑
@Component
@Slf4j
class ATask{// 创建线程池private ScheduledExecutorService scheduledExecutor = Executors.newSingleThreadScheduledExecutor();@PostConstructpublic void init(){scheduledExecutor.schedule(this::start, 10000, TimeUnit.MILLISECONDS);}public void start() {... // 原处理逻辑}
}
http://www.lryc.cn/news/113225.html

相关文章:

  • 7-数据结构-(带头节点)单链表的增删改查
  • 每天一道leetcode:剑指 Offer 53 - II. 0~n-1中缺失的数字(适合初学者二分查找)
  • 玩机搞机---安卓新机型payload.bin刷写救砖 无需专用线刷包
  • 配置固定二级子域名远程访问内网群晖NAS 7.X版 【内网穿透】——“cpolar内网穿透”
  • 【枚举】CF1706 C
  • uniapp-疫情应急管理系统学生端
  • FreeRTOS的线程间通信
  • Linux内存管理工作原理:
  • 【并发编程】ShenyuAdmin里面数据同步用到的无锁环形队列LMAX Disruptor并发框架
  • Nginx(2)
  • 二维数组的鞍点
  • go 内置函数copy()
  • Spring简述
  • 框框大学之——教育技术学
  • Android中的Apk 包体优化
  • Java基础接口详解
  • CCL 2023 电信网络诈骗案件分类评测-第一名方案
  • go test
  • 401 · 排序矩阵中的从小到大第k个数
  • 进程什么时候会进入阻塞状态
  • 炸裂,靠“吹牛”过京东一面,月薪40K
  • awk基础知识和案例
  • 重型并串式液压机械臂建模与simscape仿真
  • zookeeper的部署
  • css white-space属性
  • (十四)大数据实战——hadoop集群一键式高可用实现自动故障转移与故障初始化恢复
  • 算法-归并排序
  • 【深度学习】StyleGANv2 2019 论文,Analyzing and Improving the Image Quality of StyleGAN
  • 【AI模型系列】火力全开!百度文心3.5三大维度、20项指标国内问鼎!
  • C++ 多态性——运算符重载