之前已经说过漏桶算法,请求过来之后,由漏桶来阻挡,最后将流量漏出去,达到限流的目的,削弱峰值流量对服务器的压力。

        不过,漏桶算法真正在生产中用的并不是很多,更多的还是使用令牌桶算法。什么是令牌桶算法呢?

        说的已经很清楚了么,令牌--桶。就是一个桶里面放了令牌(一个校验或者说是一个通行证),每次有请求过来,必须要从桶里拿到令牌(许可证),才允许通过。如果拿不到,那不好意思,你就不允许通过了。

        令牌是哪里来的呢?当然是定时生成的啊,就好像桶的那个漏洞一样,一直在滴水,而这个令牌桶,也一直在往里面令牌,直到令牌桶满了,才会停止加(生成)令牌。

        那令牌桶和漏桶有啥区别呢?都是通过一个桶进行限流,漏桶似乎更方便,而且避免了生成令牌这个步骤,不是更简单么?

        这个也想了一下其中的区别,当然度娘的帮助更大!

        

java 令牌桶控制速率 令牌桶算法_线程池

 

        简单的说下,漏桶算法,其实太匀速,太平缓了,对于一些突发状况很难进行处理。

        比如,桶的大小为100,突然来了99个请求,然后又来了6个请求。

        漏桶的话,99个请求可能一直在桶里面呆着,6个请求可能就被抛弃了,当然也可以缓存起来,但是这样几乎就相当于扩大了桶的大小,总归是要有个限定的,而且,这6个请求要等待,等99个请求完全通过,才能进行处理,很容易发生超时的情况哦!

        令牌桶的话,99个请求直接就会去处理,因为令牌桶是满的(相当于是积蓄),6个请求当然无法取到足够的令牌,要等一下(缓存或者阻塞),或者是抛弃请求,不过6个令牌很快就能生成,所以稍等一会就能完成处理。

        在我看来,令牌桶就是一种有准备的方式,可以应对许多突发状况。我们知道,很多情况都是遵循二八原则,百分之二十的时间承受百分之八十的业务量。这个有准备很有必要的!

        还有就是,漏桶,一旦请求过多,要么放大空间进行缓存,要么就是抛弃请求。而令牌桶中,请求获取到令牌(通行证)之后,直接就处理了,后续请求可以阻塞,抛弃,处理部分等等方式进行处理。

        可以比漏桶,少一个桶的请求滞留!而且如果突发性间隔时间比较长的话,由于令牌桶的“有准备”,会比漏桶的速度快很多,这个就是性能方面的体现了。

写个代码简单实现下,既然是令牌,首先想到的肯定是semaphore,然后直接干就行了!

/**
 * desc: 令牌桶算法
 *
 * @author zhang
 * @date 2021/11/18 11:39
 */
public class TokenBucket {

    private static final Integer TOKEN_BUCKET = 100 ;

    private static final Integer SECOND_TOKEN = 3 ;
    // 取空两次在停止
    private static CountDownLatch countDownLatch = new CountDownLatch(1);

    public static void main(String[] args) throws InterruptedException {
        // 100 个令牌
        Semaphore semaphore = new Semaphore(TOKEN_BUCKET);
        //每秒放两个,简单处理,一个线程放
        //定时器
        ScheduledExecutorService executor = Executors.newSingleThreadScheduledExecutor();
//        semaphore.acquire(50);
        //执行
        executor.scheduleAtFixedRate(()->{
            // 查询线程个数,不能超过 TOKEN_BUCKET
            int tokenNum = semaphore.availablePermits();
            //对比,放令牌呗
            if (tokenNum < TOKEN_BUCKET){
                //释放不就是重新放进去么?嘿嘿。当然最多放 SECOND_TOKEN 个,而且不能超过桶的大小。
                semaphore.release(Math.min(SECOND_TOKEN,TOKEN_BUCKET - tokenNum));
                System.out.println("放入后令牌数量:"+tokenNum);
            }
        },0,1, TimeUnit.NANOSECONDS);

        // 来个拿令牌的吧,主线程循环啦。。算了,开个线程吧,不是啥大事。
        ExecutorService dealExecutor = Executors.newFixedThreadPool(1);
        dealExecutor.execute(()->{
            for (int i = 0; i < 10000; i++) {
                //多获取几个令牌,可以快点停止
                boolean flag = semaphore.tryAcquire(2);
                // 获取不到足够令牌,或者剩余令牌数为0的话,就结束了
                if (!flag){
                    executor.shutdownNow();
                    countDownLatch.countDown();
                    return;
                } else {
                    System.out.println(System.currentTimeMillis() + Thread.currentThread().getName()+"现在剩余令牌:" + semaphore.availablePermits());
                    if (semaphore.availablePermits() == 0){
                        executor.shutdownNow();
                        countDownLatch.countDown();
                        return;
                    }
                }
            }
        });
        countDownLatch.await();
        dealExecutor.shutdownNow();
    }
}

简单说下吧,以上代码只是为了简单描述令牌桶实现的,放入令牌桶的频率有点高哈,而取令牌更是循环一直取,令牌为0或者取不到足够令牌(2个),就会退出程序。

java 令牌桶控制速率 令牌桶算法_线程池_02

看图可能有点不对劲是吧?最后咋出来个36? 还不是程序运行的太快了。。虽然是开了单独的线程,但是写日志,是写到一起的啊,几个纳秒的差距,日志顺序自然就不对了。如果要看的更清晰一些,可以在循环取令牌的时候,可以sleep一下,每次睡眠1秒,然后放入令牌也可以,那样日志应该是按照顺序来的了。

大致思想是这样,下次看Google实现的令牌桶,封装的就完善许多了,大伙有时间也可以自己看下哈,Guava-RateLimiter !!就是这个东东哈

        no sacrifice,no victory~