当前位置: 首页 > news >正文

【Golang】用官方rate包构造简单IP限流器

文章目录

  • 使用 Go 实现基于 IP 地址的限流机制
      • 什么是 IP 限流?
    • 基于 `rate.Limiter` 实现 IP 限流
      • 1. 设计思路
      • 2. 代码实现
      • 3. 限流中间件
      • 4. 在 Gin 中使用中间件
      • 代码解释

使用 Go 实现基于 IP 地址的限流机制

在高流量的服务中,限流是一个至关重要的环节。它不仅能防止服务因过多请求而崩溃,还能保证服务的公平性。在本篇文章中,我们将介绍如何利用 Go 语言中的 golang.org/x/time/rate 包实现一个基于 IP 地址的限流机制。

什么是 IP 限流?

IP 限流是指根据客户端的 IP 地址,对每个 IP 地址发出的请求进行限制。这样可以防止某个单独的用户或机器通过频繁的请求耗尽服务器资源。

基于 rate.Limiter 实现 IP 限流

Go 语言的 golang.org/x/time/rate 包提供了一个非常实用的令牌桶限流算法,能够轻松控制请求的速率。在本实现中,我们将基于客户端 IP 地址,为每个 IP 地址创建一个独立的限流器。

1. 设计思路

我们使用 map[string]*rate.Limiter 来存储每个 IP 地址的限流器,并为每个 IP 地址设定一个令牌桶。我们还要为每个 IP 地址记录最后一次请求的时间,并在一定时间(TTL)后清除过期的限流器。

2. 代码实现

首先,我们创建一个 IPRateLimiter 结构体,该结构体维护了所有 IP 的限流器、过期时间以及并发安全的锁。

package middlewareimport ("github.com/gin-gonic/gin""github.com/sirupsen/logrus""golang.org/x/time/rate""sync""time""git.nominee.com.cn/sectrainx/stx-common/pkg/response"
)// IPRateLimiter 定义了 IP 限流器
type IPRateLimiter struct {ips        map[string]*rate.Limiter // 存储每个 IP 的限流器ipLastUsed map[string]time.Time     // 记录每个 IP 地址最后一次使用的时间mu         *sync.RWMutex            // 确保并发安全r          rate.Limit               // 每秒生成令牌数b          int                      // 令牌桶容量ttl        time.Duration            // 限流器的过期时间
}// NewIPRateLimiter 创建一个新的 IP 限流器实例
func NewIPRateLimiter(r rate.Limit, b int, ttl time.Duration) *IPRateLimiter {i := &IPRateLimiter{ips:        make(map[string]*rate.Limiter),ipLastUsed: make(map[string]time.Time),mu:         &sync.RWMutex{},r:          r,b:          b,ttl:        ttl,}return i
}// AddIP 创建并添加一个新的 IP 限流器
func (i *IPRateLimiter) AddIP(ip string) *rate.Limiter {i.mu.Lock()defer i.mu.Unlock()limiter := rate.NewLimiter(i.r, i.b)i.ips[ip] = limiteri.ipLastUsed[ip] = time.Now()	// 设置 IP 的最后使用时间return limiter
}// GetLimiter 获取指定 IP 地址的限流器
func (i *IPRateLimiter) GetLimiter(ip string) *rate.Limiter {i.mu.Lock()defer i.mu.Unlock()// 如果限流器存在,并且 IP 没有过期,则返回现有的限流器if limiter, exists := i.ips[ip]; exists {if time.Since(i.ipLastUsed[ip]) < i.ttl {i.ipLastUsed[ip] = time.Now() // 更新最后访问时间return limiter} else {// 如果限流器已经过期,删除delete(i.ips, ip)delete(i.ipLastUsed, ip)}}// 不存在,创建新的限流器return i.AddIP(ip)
}// 定时清理过期的 IP 限流器
func (i *IPRateLimiter) CleanUpExpiredLimiters() {i.mu.Lock()defer i.mu.Unlock()for ip, lastUsed := range i.ipLastUsed {if time.Since(lastUsed) > i.ttl {delete(i.ips, ip)delete(i.ipLastUsed, ip)}}
}// 启动一个定时器定期清理过期 IP 限流器
func (i *IPRateLimiter) StartCleanupRoutine() {go func() {for {time.Sleep(10 * time.Minute) // 每 10 分钟执行一次清理i.CleanUpExpiredLimiters()}}()
}

3. 限流中间件

接下来,我们实现一个 RateLimitMiddleware 中间件,用于在每次请求时检查客户端的 IP 地址,并根据限流器的状态判断是否允许请求通过。

// IPRateLimitMiddleware 根据 IP 地址进行限流
func IPRateLimitMiddleware(limiter *IPRateLimiter) gin.HandlerFunc {return func(c *gin.Context) {ip := c.ClientIP()limiter := limiter.GetLimiter(ip)// 非阻塞方式检查令牌if !limiter.Allow() {// 触发限流,返回响应xxxxreturn}c.Next()}
}

4. 在 Gin 中使用中间件

最后,我们在 Gin 路由中使用这个中间件,并定期清理过期的 IP 限流器。

package mainimport ("github.com/gin-gonic/gin""log""time""your_project/middleware" // 引入你的中间件包
)func main() {// 创建一个 IP 限流器实例,设定每秒 10 个令牌,令牌桶容量为 100,TTL 为 1 小时ipLimiter := middleware.NewIPRateLimiter(10, 100, 1*time.Hour)// 启动定时清理过期 IP 限流器ipLimiter.StartCleanupRoutine()// 创建 Gin 引擎r := gin.Default()// 将限流中间件应用到路由r.Use(middleware.RateLimitMiddleware2(ipLimiter))// 示例路由r.GET("/", func(c *gin.Context) {c.JSON(200, gin.H{"message": "Request successful!",})})// 启动服务器if err := r.Run(":8080"); err != nil {log.Fatalf("Error starting server: %v", err)}
}

代码解释

  1. IPRateLimiter 结构体:

    • ips 存储每个 IP 地址的限流器。
    • ipLastUsed 记录每个 IP 地址最后一次请求的时间。
    • rb 分别是每秒生成令牌的速率和令牌桶的容量。
    • ttl 是过期时间,表示 IP 限流器的有效期。
  2. NewIPRateLimiter 初始化一个 IPRateLimiter 实例,设置速率、容量和过期时间。

  3. GetLimiter 获取指定 IP 地址的限流器,如果限流器已经过期,会重新创建一个。

  4. CleanUpExpiredLimiters 定时清理过期的限流器。

  5. RateLimitMiddleware2 这是我们设计的限流中间件,检查请求的 IP 地址是否符合限流条件。

http://www.lryc.cn/news/605964.html

相关文章:

  • 【14】大恒相机SDK C#开发 ——Bitmap.UnlockBits()什么意思?有什么用?bmpData.Scan0;什么意思?有什么用?
  • go goroutine chan 用法
  • 网络编程(一)TCP编程和UDP编程
  • 前端工程化包管理器:从npm基础到nvm多版本管理实战
  • Vue多请求并行处理实战指南
  • Acrel-1000系列分布式光伏监控系统在湖北荆门一马光彩大市场屋顶光伏发电项目中应用
  • 【人工智能-15】OpenCV直方图均衡化,模板匹配,霍夫变换,图像亮度变换,形态学变换
  • webpack-babel
  • ESXI虚拟交换机 + H3C S5120交换机 + GR5200路由器组网笔记
  • 如何将照片从 realme 手机传输到电脑?
  • 使用橙武低代码平台构建摄影店管理系统的完整指南
  • 【爬虫实战】使用Python和JS逆向基于webpack的游戏平台
  • Rust × WebAssembly 项目脚手架详解
  • Kubernetes 应用部署实战:为什么需要 Kubernetes?
  • 本土发货模式兴起,如何选择合适的海外仓系统?
  • 单张卡牌类
  • 星云能量传送特效技术详解
  • Servlet修改新增思路
  • C语言---结构体(格式、用法、嵌套、初始化)、共用体、枚举类型、typedef类型
  • 舱驾操作系统架构规划
  • 使用的IDE没有内置MCP客户端怎么办?
  • AI 类型的 IDE
  • AI IDE+AI 辅助编程-生成的大纲-一般般
  • 掩码语言模型(MLM)技术解析:理论基础、演进脉络与应用创新
  • 从循环依赖谈 Chromium 模块化设计:编译结构与最佳实践
  • 基于 Amazon Nova Sonic 和 MCP 构建语音交互 Agent
  • 开发避坑短篇(11):Oracle DATE(7)到MySQL时间类型精度冲突解决方案
  • USRP捕获手机/路由器数据传输信号波形(下)
  • 6.苹果ios逆向-过ssl证书检测-安装SSL Kill Switch 3
  • JVM字节码文件结构剖析