限流之漏桶算法

java 复制代码
package com.gewb.test;

import java.time.LocalDateTime;
import java.time.format.DateTimeFormatter;
import java.util.concurrent.*;
import java.util.concurrent.atomic.AtomicInteger;

/**
 * @Author: James.Ge
 * @Date: 2025/12/28
 * @Description 漏桶算法
 */
public class LeakyBucketRateLimiter {
    /**
     * 漏桶容量
     */
    private final int bucketCapacity;

    /**
     * 请求处理速率
     */
    private final int consumeRate;

    /**
     * 请求队列
     */
    private final BlockingQueue<Runnable> bucket;

    private final ScheduledExecutorService executorService;

    public LeakyBucketRateLimiter(int bucketCapacity, int consumeRate) {
        this.bucketCapacity = bucketCapacity;
        this.consumeRate = consumeRate;
        this.bucket = new LinkedBlockingQueue<>(bucketCapacity);
        this.executorService = Executors.newScheduledThreadPool(1);
        // 启动定期消费请求的线程
        startConsuming();
    }

    private void startConsuming() {
//        ScheduledExecutorService executorService = Executors.newScheduledThreadPool(1);
        executorService.scheduleAtFixedRate(() -> {
            for (int i = 0; i < consumeRate; i++) {
                Runnable poll = bucket.poll();
                if(poll != null) {
                    poll.run();
                }
            }
        }, 0, 1, TimeUnit.SECONDS);
    }

    public boolean addRequest(Runnable request) {
        if(bucket.offer(request)) {
            System.out.println("请求已经添加到桶");
            return true;
        } else {
            System.out.println("请求被丢弃,桶已满");
            return false;
        }
    }

    /**
     * 关闭限流器,释放资源
     */
    public void shutdown() {
        while(!bucket.isEmpty()) {
            try {
                Thread.sleep(100);
            } catch (InterruptedException e) {
                throw new RuntimeException(e);
            }
        }
        executorService.shutdown();

            // 在异常时可保证能够正常释放资源
//            if (!executorService.awaitTermination(5, TimeUnit.SECONDS)) {
//                executorService.shutdownNow();
//            }
//        } catch (InterruptedException e) {
//            executorService.shutdownNow();
//            Thread.currentThread().interrupt();
//        }
        }


        public static void main(String[] args) {
        LeakyBucketRateLimiter rateLimiter = new LeakyBucketRateLimiter(5, 2);
        for (int i = 0; i < 10; i++) {
            int finalI = i;
            rateLimiter.addRequest(() -> {
                LocalDateTime now = LocalDateTime.now();
                String format = now.format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss.SSS"));
                System.out.println("处理第" + finalI + "个请求:" + format);
            });
        }

//        try {
//            Thread.sleep(5000);
//        } catch (InterruptedException e) {
//            Thread.currentThread().interrupt();
//        }
        rateLimiter.shutdown();
        System.out.println("程序结束");
    }
}

运行结果:

"C:\Program Files\Java\jdk-17\bin\java.exe" "-javaagent:D:\JetBrains\IntelliJ IDEA 2023.1.2\lib\idea_rt.jar=63321:D:\JetBrains\IntelliJ IDEA 2023.1.2\bin" -Dfile.encoding=UTF-8 -classpath D:\ideaWorkspace2023.1\jdk17\out\production\jdk17;D:\ideaWorkspace2023.1\jdk17\lib\caffeine-3.2.0.jar;D:\ideaWorkspace2023.1\jdk17\lib\jspecify-1.0.0.jar;D:\ideaWorkspace2023.1\jdk17\lib\error_prone_annotations-2.36.0.jar com.gewb.test.LeakyBucketRateLimiter

请求已经添加到桶

请求已经添加到桶

请求已经添加到桶

请求已经添加到桶

请求已经添加到桶

请求已经添加到桶

请求被丢弃,桶已满

请求被丢弃,桶已满

请求被丢弃,桶已满

请求被丢弃,桶已满

处理第0个请求:2025-12-29 11:37:58.659

处理第1个请求:2025-12-29 11:37:58.680

处理第2个请求:2025-12-29 11:37:59.664

处理第3个请求:2025-12-29 11:37:59.664

处理第4个请求:2025-12-29 11:38:00.662

处理第5个请求:2025-12-29 11:38:00.662

程序结束

Process finished with exit code 0

相关推荐
君义_noip1 小时前
信息学奥赛一本通 1661:有趣的数列 | 洛谷 P3200 [HNOI2009] 有趣的数列
c++·算法·组合数学·信息学奥赛·csp-s
leaves falling1 小时前
C语言内存函数-
c语言·开发语言
程序员:钧念1 小时前
深度学习与强化学习的区别
人工智能·python·深度学习·算法·transformer·rag
英英_2 小时前
MATLAB数值计算基础教程
数据结构·算法·matlab
一起养小猫3 小时前
LeetCode100天Day14-轮转数组与买卖股票最佳时机
算法·leetcode·职场和发展
至为芯3 小时前
IP6537至为芯支持双C口快充输出的45W降压SOC芯片
c语言·开发语言
hele_two3 小时前
快速幂算法
c++·python·算法
kk哥88993 小时前
如何快速掌握JavaSE的核心语法?
java
我是一只小青蛙8883 小时前
AVL树:平衡二叉搜索树原理与C++实战
java·jvm·面试
小羊羊Python3 小时前
SoundMaze v1.0.1正式发布!
开发语言·c++