@Scheduled两个整点任务:触发必为整点,执行时间藏着线程池玄机

在同一个类中,两个标注@Scheduled且均配置为"整点执行"的方法,触发时间一定是整点,但实际执行的"开始时间"可能受线程池影响出现延迟,核心结论如下:

1. 触发时间:严格按配置执行(必为整点)

@Scheduled的底层调度器(如ThreadPoolTaskScheduler)会严格按照时间规则(如cron = "0 0 * * * ?")在每个整点触发任务。无论两个方法是否在同一个类,调度器都会在00:00、1:00、2:00... 同时向这两个任务发出"执行信号"。

这一步由Spring的定时任务框架保证,与类结构、方法数量无关。

2. 执行时间:可能延迟,取决于线程池配置

任务"触发"后,实际开始执行的时间由调度器的线程池决定:

(1)默认单线程配置(poolSize = 1

Spring默认的ThreadPoolTaskScheduler线程池大小为1,此时两个任务会串行执行

  • 第一个任务先占用线程开始执行;
  • 第二个任务需等待第一个任务执行完毕后,才能获取线程开始执行。

示例

java 复制代码
@Scheduled(cron = "0 0 * * * ?")
public void task1() {
    System.out.println("task1 开始:" + LocalDateTime.now()); // 00:00:00
    try { Thread.sleep(5000); } catch (InterruptedException e) {} // 耗时5秒
}

@Scheduled(cron = "0 0 * * * ?")
public void task2() {
    System.out.println("task2 开始:" + LocalDateTime.now()); // 00:00:05(延迟5秒)
}

结果:两个任务都在00:00被触发,但task2因等待线程,实际开始时间延迟至00:00:05。

(2)多线程配置(poolSize ≥ 2

若手动配置线程池大小≥2,两个任务可并行执行,同时在整点开始:

java 复制代码
// 配置线程池
@Bean
public TaskScheduler taskScheduler() {
    ThreadPoolTaskScheduler scheduler = new ThreadPoolTaskScheduler();
    scheduler.setPoolSize(2); // 2个线程
    return scheduler;
}

// 任务执行结果:
// task1 开始:00:00:00
// task2 开始:00:00:00

3. 关键结论

  • 触发时间 :必为整点(由@Scheduled的时间规则严格保证)。
  • 执行时间
    • 单线程:后执行的任务会延迟(等待前一个任务释放线程)。
    • 多线程:若线程池有足够空闲线程,两个任务会在整点同时开始执行。

若业务要求"必须在整点同时执行",需配置足够的线程池大小(poolSize ≥ 任务数量)。

相关推荐
南囝coding4 分钟前
Vercel 发布 AI Gateway 神器!可一键访问数百个模型,助力零门槛开发 AI 应用
前端·后端
耀耀_很无聊15 分钟前
14_Spring Boot 跨域(CORS)处理指南
spring boot·后端
他日若遂凌云志24 分钟前
深入拆解 Windows Socket 五种 I/O 模型:核心机制、Linux 差异与场景适配
后端
小码编匠24 分钟前
开箱即用!集成 YOLO+OpenCV+OCR 的 WebAI 平台(支持RTSP/RTMP视频流识别与自训练)
spring boot·后端·opencv
zcychong24 分钟前
如何让A、B、C三个线程按严格顺序执行(附十一种解)?
java·面试
文心快码BaiduComate1 小时前
再获殊荣!文心快码荣膺2025年度优秀软件产品!
前端·后端·代码规范
天天摸鱼的java工程师1 小时前
SpringBoot + RabbitMQ + Redis + MySQL:社交平台私信发送、已读状态同步与历史消息缓存
java·后端
Kiri霧1 小时前
Rust数组与向量
开发语言·后端·rust
特立独行的猫a1 小时前
Rust语言入门难,难在哪?所有权、借用检查器、生命周期和泛型介绍
开发语言·后端·rust
间彧1 小时前
Spring Boot Actuator详解:生产级监控与管理工具
后端