Golang | 令牌桶限流算法

  • 限流算法的目的是控制对后端接口的访问频率,防止因过度访问导致系统崩溃。
  • 网站登录接口是限流的典型例子,爬虫或恶意用户可能疯狂调用登录接口,导致数据库压力过大。
  • 通过限制接口的QPS(每秒查询率),可以保护后端数据库不受冲击。
  • 令牌桶算法是一种经典的限流算法,将请求想象成令牌,桶以一定速度放入令牌。
  • 桶的容量决定瞬间最多能满足的请求数,生产速度代表平均供应速度。
  • 谷歌官方提供了基于令牌桶算法的实现,位于guava库的x包下。
  • 使用rate.newLimit方法创建限流器,指定时间间隔和桶容量。
  • 限流器接受三种取令牌方式:wait、allow和reserve。
  • wait方式会等待桶内令牌足够为止;allow方式立即返回是否允许取走令牌;reserve方式会计算等待时间。

go 复制代码
package tests

import (
	"sync/atomic"
	"time"
	"golang.org/x/time/rate"
)

// TotalQuery 记录请求总数
var TotalQuery int32

// Handler 模拟接口
func Handler() {
	atomic.AddInt32(&TotalQuery, 1)
	time.Sleep(50 * time.Millisecond)
}

func CallHandler() {
	// 每隔100ms生成一个令牌,最大QPS限制为10
	limiter := rate.NewLimiter(rate.Every(100*time.Millisecond), 10)
	for {
		// 方式一:WaitN()
		//ctx, cancel := context.WithTimeout(context.Background(), 100*time.Millisecond)
		//defer cancel()
		//limiter.WaitN(ctx, 1) // 阻塞,直到桶中有N个令牌。N=1时等价于Wait(ctx)
		//Handler()

		// 方式二:AllowN()
		// 不等,当前桶中是否至少还有N个令牌,如果有则返回true。N=1时等价于Allow(time.Time)
		//if limiter.AllowN(time.Now(), 1) {
		//	Handler()
		//}

		// 方式三:ReserveN()
		// reserve.Delay()告诉你还需要等多久才会有充足的令牌,等待相应时间,再执行
		reserve := limiter.ReserveN(time.Now(), 1)
		time.Sleep(reserve.Delay())
		Handler()
	}
}
  • 令牌桶(Token Bucket)算法是一种常用的限流算法,常用于网络流量控制和高并发服务的请求限速。它本质上是一个允许突发请求但控制平均速率的机制,兼具了平滑输出和灵活突发的优点。
  • 令牌桶算法通过一个"桶"来存储"令牌",系统按照固定速率向桶中添加令牌,处理请求时必须"取走"一个令牌,如果桶里没有令牌了,请求就被拒绝或延迟处理。
  • 每次处理请求时:
    • 令牌补充:根据当前时间与上次补充时间的差值,计算应添加的令牌数量:tokens += (now - last_refill_time) * r
    • 令牌溢出处理:如果 tokens > b,令牌数取为桶最大容量。
    • 判断请求是否能通过:
      • 如果 tokens >= 1,扣除一个令牌,允许请求通过。
      • 否则,拒绝请求或阻塞等待。
相关推荐
微信公众号:AI创造财富16 分钟前
conda create -n modelscope python=3.8 conda: command not found
开发语言·python·conda
鱼会上树cy16 分钟前
空间解析几何10:三维圆弧拟合【附MATLAB代码】
开发语言·matlab
IT艺术家-rookie1 小时前
golang--channel的关键特性和行为
开发语言·后端·golang
Cyrus_柯3 小时前
C++(面向对象编程——关键字)
开发语言·c++·算法·面向对象
大龄Python青年3 小时前
C语言 函数怎样通过数组来返回多个值
c语言·开发语言
LQYYDSY3 小时前
【C语言极简自学笔记】重讲运算符
c语言·开发语言·笔记
2013编程爱好者3 小时前
C++二分查找
开发语言·c++·算法·二分查找
电商数据girl3 小时前
【经验分享】浅谈京东商品SKU接口的技术实现原理
java·开发语言·前端·数据库·经验分享·eclipse·json
十五年专注C++开发4 小时前
QSimpleUpdater:解锁 Qt 应用自动更新的全新姿势
开发语言·c++·qt
小猫咪怎么会有坏心思呢4 小时前
华为OD机考-货币单位换算-字符串(JAVA 2025B卷)
java·开发语言·华为od