定时任务线程池-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();
 }
相关推荐
咖啡八杯32 分钟前
GoF设计模式——中介者模式
java·后端·spring·设计模式
青石路4 小时前
记一次多JDK版本问题的排查,一坑套一坑,差点没爬上来
java
像我这样帅的人丶你还7 小时前
Java 后端详解(五):Redis 缓存
java·后端·全栈
plainGeekDev9 小时前
GreenDAO → Room
android·java·kotlin
亦暖筑序14 小时前
Java 8老系统AI Workflow实战:把一次性AI对话升级成可恢复工作流
java·后端
敲代码的彭于晏15 小时前
Bean 生命周期完全图解:前端同学也能看懂的 Spring 核心机制
java·前端·后端
plainGeekDev16 小时前
ButterKnife → ViewBinding
android·java·kotlin
像我这样帅的人丶你还1 天前
Java 后端详解(四):分页与搜索
java·javascript·后端
她的男孩1 天前
数据权限为什么不能只靠注解?Forge 的 Mapper 层 SQL 改写源码拆解
java·后端·架构
tntxia1 天前
Mybatis的日志输入
java