定时任务线程池-scheduleAtFixedRate和scheduleWithFixedDelay

定时任务线程池-scheduleAtFixedRate和scheduleWithFixedDelay

  • 前言
  • 一、核心区别
  • 二、源码处看区别
    • [2.1 提交任务的源码区别](#2.1 提交任务的源码区别)
    • [2.2 任务执行后的区别](#2.2 任务执行后的区别)
      • [2.2.1 ScheduledFutureTask 的 run](#2.2.1 ScheduledFutureTask 的 run)
      • [2.2.2 setNextRunTime](#2.2.2 setNextRunTime)
  • 三、代码示例
    • [3.1 scheduleAtFixedRate 执行时间大于间隔](#3.1 scheduleAtFixedRate 执行时间大于间隔)

前言

本章节介绍定时任务线程池中 scheduleAtFixedRatescheduleWithFixedDelay 的区别


一、核心区别

  1. scheduleAtFixedRate 是固定的频率执行,如果任务执行时间过程超过了固定频率需要注意不会并行执行,而是上个任务结束后立即执行下个任务
  2. scheduleWithFixedDelay 是固定的延迟执行,上一个任务执行完成后多长时间再次执行

二、源码处看区别

2.1 提交任务的源码区别

2.2 任务执行后的区别

ScheduledFutureTask 是最终提交给线程池的任务类,肯定实现了 Runnable 或者 Callable,其实是 Runnable,所以任务执行的时候肯定会执行它的 run()

2.2.1 ScheduledFutureTask 的 run

java 复制代码
public void run() {
if (!canRunInCurrentRunState(this))
         cancel(false);
     else if (!isPeriodic())
         super.run();
     // super.runAndReset 这里面会执行你提交的任务
     // 注意这里是同步的, 也就是说 setNextRunTime 是在当前任务结束后才会执行, 因此同一个任务不会并行执行
     else if (super.runAndReset()) {
     		// 设置下次执行时间
         setNextRunTime();
         reExecutePeriodic(outerTask);
     }
 }

2.2.2 setNextRunTime

java 复制代码
private void setNextRunTime() {
     long p = period;
     // 如果 period > 0, 是上次开始执行时间+间隔
     if (p > 0)
         time += p;
     else
     	// 如果 period<0, 则是需要重新计算下次执行时间
      time = triggerTime(-p);
 }

三、代码示例

3.1 scheduleAtFixedRate 执行时间大于间隔

java 复制代码
@Test
public void test2() {
     ScheduledExecutorService scheduledExecutorService = Executors.newScheduledThreadPool(2);
     AtomicInteger counter = new AtomicInteger(1);
     ScheduledFuture<?> scheduledFuture = scheduledExecutorService.scheduleAtFixedRate(() -> {
         int currentNumber = counter.getAndIncrement();
         // 这里日志要打印时间戳, 否则没有效果
         log.info("AtFixed - start - {}", currentNumber);
         ThreadUtil.sleep(5L, TimeUnit.SECONDS);
         log.info("AtFixed - end - {}", currentNumber);
     }, 0, 2, TimeUnit.SECONDS);

     ThreadUtil.waiting();
 }
相关推荐
Seven9714 小时前
一致性Hash算法:如何实现分布式系统中的高效数据分片?
java
摇滚侠14 小时前
IDEA 生成 try catch 快捷键
java·ide·intellij-idea
阿旭超级学得完15 小时前
C++11包装器(function和bind)
java·开发语言·c++·算法·哈希算法·散列表
掉鱼的猫16 小时前
Spring AI 2.0 GA 倒计时:先别急,来看看 Java AI 框架的另一条路
java·openai·agent
Refrain_zc16 小时前
Android 应用内 APK 安装全方案:从静默安装到普通安装的详解
java
正儿八经的少年16 小时前
Spring Boot 两种激活配置方式的作用与区别
java·spring boot·后端
云烟成雨TD16 小时前
Spring AI Alibaba 1.x 系列【52】Interrupts 中断机制:节点执行前后静态中断
java·人工智能·spring
疯狂成瘾者16 小时前
Spring Boot 项目中的 SMTP 邮件验证码服务技术解析
java·spring boot·后端
y = xⁿ16 小时前
Java并发八股学习日记
java·开发语言·学习