偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

別再依賴重型調(diào)度框架!Spring Boot + 時間輪算法打造輕量級分布式定時任務

開發(fā) 前端
時間輪算法原本是操作系統(tǒng)底層的經(jīng)典設(shè)計,如今在 Spring Boot 場景下被重新激活。它用極低的成本突破了傳統(tǒng)定時任務的性能瓶頸,讓普通應用也能輕松應對百萬級調(diào)度場景。

在分布式系統(tǒng)中,定時任務是核心組成部分:訂單超時、會話管理、緩存過期、鎖釋放……這些場景無處不在。很多團隊習慣依賴 Quartz、ElasticJob 等重量級調(diào)度框架,雖然功能齊全,但復雜性和運維成本也隨之而來。另一方面,ScheduledExecutorService 或 @Scheduled 雖然簡單,但當調(diào)度規(guī)模上升到 百萬級任務 時,就會遭遇嚴重的資源浪費與性能瓶頸。

有沒有一種方式,既能像 @Scheduled 一樣輕量易用,又能支撐百萬級任務調(diào)度?答案就是 時間輪(Timing Wheel)算法。本文將帶你基于 Spring Boot + 時間輪 打造一個輕量、高效、可擴展的定時任務引擎,實現(xiàn)單機百萬級任務調(diào)度的能力,并探索分布式場景下的應用方案。

傳統(tǒng)定時任務的局限性

常見寫法

@Scheduled(fixedRate = 5000)
public void checkOrderStatus() {
    // 每5秒檢查訂單狀態(tài)
}

這種方式雖然直觀,但存在以下問題:

  • 線程資源浪費:每個任務通常會綁定獨立線程。
  • 精度不足:最小調(diào)度間隔常常大于等于 10ms。
  • 擴展性差:調(diào)度 10 萬任務意味著要分配 10 萬線程。
  • 內(nèi)存占用高:每個線程大約消耗 1MB ??臻g。

性能對比測試

實際壓力測試表明,基于線程池的定時方案在百萬任務級別幾乎不可用,而時間輪算法的內(nèi)存占用與執(zhí)行延遲則保持穩(wěn)定。

時間輪算法核心解析

 基本原理

時間輪是一種環(huán)形數(shù)組,每個槽位(slot)代表一段時間間隔:

  • tickDuration:槽位跨度(如 1ms)。
  • ticksPerWheel:槽位數(shù)量(如 512)。
  • 指針:周期性移動,每次觸發(fā)槽位內(nèi)的任務。

直觀圖示:

┌─────────────── 時間輪(環(huán)形結(jié)構(gòu)) ───────────────┐
   │                                                 │
   │   [0] → [1] → [2] → ... → [510] → [511] ┐       │
   │     ↑                                    │       │
   │     └────────────── 當前指針 ─────────────┘       │
   │                                                 │
   └─────────────────────────────────────────────────┘


- 時間被劃分為固定間隔 tickDuration(例如 1ms)
- 指針每次跳動處理一個槽位
- 槽位中可能包含多個任務,執(zhí)行到期任務并重新調(diào)度剩余的

當延遲時間超過一個輪盤周期時,可以引入 分層時間輪,通過類似“進位”的方式處理長延時任務,避免數(shù)組過大。

Spring Boot 集成時間輪

引入依賴

<dependency>
    <groupId>io.netty</groupId>
    <artifactId>netty-common</artifactId>
    <version>4.1.94.Final</version>
</dependency>

時間輪封裝類

//src/main/java/com/icoderoad/scheduler/HashedWheelScheduler.java
package com.icoderoad.scheduler;


import io.netty.util.HashedWheelTimer;
import io.netty.util.Timeout;
import io.netty.util.TimerTask;
import org.springframework.stereotype.Component;


import jakarta.annotation.PreDestroy;
import java.util.concurrent.TimeUnit;


@Component
public class HashedWheelScheduler {
    private final HashedWheelTimer timer;


    public HashedWheelScheduler() {
        // 創(chuàng)建時間輪:1ms 精度,512 個槽位
        this.timer = new HashedWheelTimer(
                Thread::new,
                1,
                TimeUnit.MILLISECONDS,
                512
        );
    }


    // 一次性任務
    public Timeout schedule(Runnable task, long delay, TimeUnit unit) {
        return timer.newTimeout(timeout -> task.run(), delay, unit);
    }


    // 固定頻率任務
    public void scheduleAtFixedRate(Runnable task, long initialDelay, long period, TimeUnit unit) {
        timer.newTimeout(new TimerTask() {
            @Override
            public void run(Timeout timeout) {
                task.run();
                timer.newTimeout(this, period, unit);
            }
        }, initialDelay, unit);
    }


    @PreDestroy
    public void shutdown() {
        timer.stop();
    }
}

業(yè)務集成

//src/main/java/com/icoderoad/service/OrderService.java
package com.icoderoad.service;


import com.icoderoad.scheduler.HashedWheelScheduler;
import io.netty.util.Timeout;
import org.springframework.stereotype.Service;


import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.TimeUnit;


@Service
public class OrderService {
    private final HashedWheelScheduler scheduler;
    private final Map<Long, Timeout> timeoutMap = new ConcurrentHashMap<>();


    public OrderService(HashedWheelScheduler scheduler) {
        this.scheduler = scheduler;
    }


    // 創(chuàng)建訂單時,設(shè)置 30 分鐘未支付自動取消
    public void createOrder(Order order) {
        Timeout timeout = scheduler.schedule(
                () -> cancelUnpaidOrder(order.getId()),
                30,
                TimeUnit.MINUTES
        );
        timeoutMap.put(order.getId(), timeout);
    }


    // 支付成功,取消任務
    public void orderPaid(Long orderId) {
        Timeout timeout = timeoutMap.remove(orderId);
        if (timeout != null) {
            timeout.cancel();
        }
    }


    private void cancelUnpaidOrder(Long orderId) {
        timeoutMap.remove(orderId);
        // 執(zhí)行訂單取消邏輯
    }
}

高級能力拓展

  • 分布式調(diào)度:結(jié)合 Redis 實現(xiàn)跨節(jié)點協(xié)調(diào)。
  • 任務持久化:將任務快照存入數(shù)據(jù)庫,支持宕機恢復。
  • 運行監(jiān)控:暴露 REST API,獲取任務指標與監(jiān)控信息。

性能優(yōu)化策略

  • 參數(shù)調(diào)優(yōu):根據(jù) CPU 核數(shù)動態(tài)調(diào)整 tick 與槽位數(shù)量。
  • 任務合并:將批量小任務合并成單個批處理任務,降低調(diào)度開銷。
  • 限流防雪崩:通過信號量控制并發(fā),避免瞬時任務過載。

典型應用案例

  • 電商系統(tǒng):訂單支付超時關(guān)閉。
  • 金融交易:債券 T+1 結(jié)算。
  • 游戲服務器:技能冷卻計時。
  • 緩存管理:數(shù)據(jù)過期與鎖自動釋放。

生產(chǎn)部署方案

  • 高可用架構(gòu):多節(jié)點部署 + 分布式鎖。
  • 配置優(yōu)化
timing-wheel:
  tick-duration: 1ms
  wheel-size: 512
  worker-threads: 4
  max-pending: 1000000
  recovery:
    enabled: true
    interval: 30s

源碼實現(xiàn)揭秘

簡化版時間輪實現(xiàn)

package com.icoderoad.scheduler;


import java.util.Queue;
import java.util.concurrent.ConcurrentLinkedQueue;
import java.util.concurrent.locks.LockSupport;


public class SimpleHashedWheelTimer {
    private final long tickDuration;              // 每個槽位的時間間隔(ms)
    private final HashedWheelBucket[] wheel;      // 時間輪的槽位數(shù)組
    private volatile int tick;                    // 當前指針位置


    public SimpleHashedWheelTimer(int ticksPerWheel, long tickDuration) {
        this.tickDuration = tickDuration;
        this.wheel = new HashedWheelBucket[ticksPerWheel];
        for (int i = 0; i < ticksPerWheel; i++) {
            wheel[i] = new HashedWheelBucket();
        }
        new Thread(this::run).start();            // 啟動后臺線程
    }


    private void run() {
        long startTime = System.nanoTime();
        while (true) {
            long deadline = startTime + (tick + 1) * tickDuration * 1_000_000;
            long currentTime = System.nanoTime();
            if (currentTime < deadline) {
                LockSupport.parkNanos(deadline - currentTime); // 等待下一個 tick
                continue;
            }
            int idx = tick & (wheel.length - 1);  // 計算當前槽位下標
            wheel[idx].expireTimeouts();          // 執(zhí)行到期任務
            tick++;                               // 指針前進一格
        }
    }


    // 新增任務
    public void newTimeout(Runnable task, long delay) {
        long deadline = System.nanoTime() + delay * 1_000_000;
        int ticks = (int) (delay / tickDuration);
        int stopIndex = (tick + ticks) & (wheel.length - 1);
        wheel[stopIndex].addTimeout(new TimeoutTask(task, deadline));
    }


    // 槽位結(jié)構(gòu)
    static class HashedWheelBucket {
        private final Queue<TimeoutTask> tasks = new ConcurrentLinkedQueue<>();
        void addTimeout(TimeoutTask task) {
            tasks.offer(task);
        }
        void expireTimeouts() {
            while (!tasks.isEmpty()) {
                TimeoutTask task = tasks.poll();
                if (task.deadline <= System.nanoTime()) {
                    task.run();   // 到期任務執(zhí)行
                } else {
                    // 未到期的任務可以重新計算位置放回
                }
            }
        }
    }


    // 任務包裝類
    static class TimeoutTask implements Runnable {
        final Runnable task;
        final long deadline;
        TimeoutTask(Runnable task, long deadline) {
            this.task = task;
            this.deadline = deadline;
        }
        @Override
        public void run() {
            task.run();
        }
    }
}

 逐行解釋

  • tickDuration:時間粒度,例如 1ms。
  • wheel[]:環(huán)形數(shù)組,每個元素是一個 槽位桶,用隊列存放任務。
  • tick:當前指針,每次加 1 表示時間輪前進一步。
  • newTimeout:計算任務要落入的槽位,并插入隊列。
  • run():后臺線程循環(huán)運行:

計算下一次 tick 的觸發(fā)時間。

等待到達時間。

執(zhí)行當前槽位的到期任務。

時間輪工作流圖

時間流逝 →
┌───────────────────────────────────────┐
│ tick=0   tick=1   tick=2  ... tick=n │
│ [槽0] →  [槽1] →  [槽2] → ... [槽n]  │
└───────────────────────────────────────┘
              ↑
       指針移動到此槽位時,執(zhí)行到期任務

這套機制本質(zhì)上是 延遲隊列的環(huán)形優(yōu)化,相比傳統(tǒng)堆結(jié)構(gòu)定時器(如優(yōu)先隊列),在百萬任務規(guī)模下能顯著減少內(nèi)存消耗與 CPU 計算成本。

總結(jié)與展望

方案優(yōu)勢:

  • 極致性能:單機支撐百萬任務。
  • 毫秒精度:滿足金融、游戲等高精度場景。
  • 資源節(jié)省:單線程即可支撐大規(guī)模任務。
  • 無縫融合:與 Spring Boot 配合自然。

適用場景:

  • 大規(guī)模延遲任務(電商、會話管理)。
  • 高精度定時任務(金融交易、技能冷卻)。
  • 邊緣計算、物聯(lián)網(wǎng)等資源受限環(huán)境。

未來演進方向:

  • 分布式時間輪:跨節(jié)點調(diào)度與負載均衡。
  • 持久化增強:任務快照與快速恢復。
  • 動態(tài)調(diào)優(yōu):運行時修改 tick 與槽位。
  • 智能調(diào)度:基于歷史數(shù)據(jù)的 AI 優(yōu)化。

時間輪算法原本是操作系統(tǒng)底層的經(jīng)典設(shè)計,如今在 Spring Boot 場景下被重新激活。它用極低的成本突破了傳統(tǒng)定時任務的性能瓶頸,讓普通應用也能輕松應對百萬級調(diào)度場景。對于追求高性能與低資源消耗的團隊而言,這無疑是定時任務領(lǐng)域的秘密武器。

責任編輯:武曉燕 來源: 路條編程
相關(guān)推薦

2025-08-01 08:47:45

2025-07-28 01:12:00

2024-02-19 00:00:00

分布式定時任務框架

2022-08-09 08:40:37

框架分布式定時任務

2022-03-23 11:45:39

Quartz數(shù)據(jù)庫節(jié)點

2023-06-26 00:14:28

Openjob分布式任務

2022-03-28 07:51:25

分布式定時任務

2024-04-29 08:42:23

2019-11-15 10:16:27

分布式任務框架

2023-01-04 09:23:58

2022-03-17 09:55:05

架構(gòu)分布式選型

2023-12-18 10:24:59

2020-09-29 19:20:05

鴻蒙

2025-07-08 02:12:00

2022-03-07 11:20:01

分布式代碼微服務

2025-08-26 04:00:00

2019-11-12 09:32:39

分布式elastic-job分片

2023-05-08 16:38:46

任務調(diào)度分布式任務調(diào)度

2020-11-06 12:12:35

HarmonyOS

2015-06-17 14:10:34

Redis分布式系統(tǒng)協(xié)調(diào)
點贊
收藏

51CTO技術(shù)棧公眾號