速率限制是 Web 开发和 API 设计中的一个关键概念。它确保用户或系统在特定时间范围内只能向服务器发出有限数量的请求。在这篇博文中,我们将探讨什么是速率限制、为什么它很重要,以及如何在 Go 中实现一个简单的速率限制器。
什么是速率限制?
想象一个主题公园,里面有过山车,每 10 分钟只能容纳 10 人。如果超过 10 人试图在该时间内上车,他们将不得不等待。这个类比反映了软件系统中速率限制的原理。
用技术术语来说,速率限制限制客户端(例如用户、设备或 IP 地址)在预定义时间内可以发送到服务器的请求数量。它有帮助:
- 防止滥用并确保公平使用资源。
- 保护服务器不被过多的流量淹没。
- 避免过度使用第三方 API 或服务,造成高昂代价。
例如,API 可能允许每个用户每分钟 100 个请求。如果用户超出此限制,服务器将拒绝进一步的请求,直到限制重置。
速率限制如何运作?
实现速率限制的一种常见方法是通过令牌桶算法。其工作原理如下:
- 存储桶以固定数量的令牌(例如 10)开始。
- 每个请求都会从存储桶中删除一个令牌。
- 如果桶中没有剩余令牌,则请求被拒绝。
- 代币会以稳定的速度(例如每秒 1 个代币)补充,直到桶装满。
在 Go 中构建一个简单的速率限制器
让我们深入研究在 Go 中构建一个速率限制器,将每个客户端限制为每分钟 3 个请求。
第 1 步:定义速率限制器结构
我们将使用sync.Mutex来确保线程安全并存储令牌数量、最大容量和填充率等信息。
package main import ( "sync" "time" ) type RateLimiter struct { tokens float64 // Current number of tokens maxTokens float64 // Maximum tokens allowed refillRate float64 // Tokens added per second lastRefillTime time.Time // Last time tokens were refilled mutex sync.Mutex } func NewRateLimiter(maxTokens, refillRate float64) *RateLimiter { return &RateLimiter{ tokens: maxTokens, maxTokens: maxTokens, refillRate: refillRate, lastRefillTime: time.Now(), } }
第 2 步:实施令牌补充逻辑
代币应根据上次充值以来的经过时间定期补充。
func (r *RateLimiter) refillTokens() { now := time.Now() duration := now.Sub(r.lastRefillTime).Seconds() tokensToAdd := duration * r.refillRate r.tokens += tokensToAdd if r.tokens > r.maxTokens { r.tokens = r.maxTokens } r.lastRefillTime = now }
第 3 步:检查请求是否被允许
Allow 方法将根据可用令牌确定请求是否可以继续。
func (r *RateLimiter) Allow() bool { r.mutex.Lock() defer r.mutex.Unlock() r.refillTokens() if r.tokens >= 1 { r.tokens-- return true } return false }
第 4 步:对每个 IP 应用速率限制
为了限制每个客户端的请求,我们将创建 IP 地址到各自速率限制器的映射。
type IPRateLimiter struct { limiters map[string]*RateLimiter mutex sync.Mutex } func NewIPRateLimiter() *IPRateLimiter { return &IPRateLimiter{ limiters: make(map[string]*RateLimiter), } } func (i *IPRateLimiter) GetLimiter(ip string) *RateLimiter { i.mutex.Lock() defer i.mutex.Unlock() limiter, exists := i.limiters[ip] if !exists { // Allow 3 requests per minute limiter = NewRateLimiter(3, 0.05) i.limiters[ip] = limiter } return limiter }
第 5 步:创建用于速率限制的中间件
最后,我们将创建一个 HTTP 中间件,对每个客户端实施速率限制。
package main import ( "sync" "time" ) type RateLimiter struct { tokens float64 // Current number of tokens maxTokens float64 // Maximum tokens allowed refillRate float64 // Tokens added per second lastRefillTime time.Time // Last time tokens were refilled mutex sync.Mutex } func NewRateLimiter(maxTokens, refillRate float64) *RateLimiter { return &RateLimiter{ tokens: maxTokens, maxTokens: maxTokens, refillRate: refillRate, lastRefillTime: time.Now(), } }
第 6 步:设置服务器
以下是如何将它们连接在一起并测试速率限制器。
func (r *RateLimiter) refillTokens() { now := time.Now() duration := now.Sub(r.lastRefillTime).Seconds() tokensToAdd := duration * r.refillRate r.tokens += tokensToAdd if r.tokens > r.maxTokens { r.tokens = r.maxTokens } r.lastRefillTime = now }
测试速率限制器
启动服务器并使用curl或浏览器进行测试:
func (r *RateLimiter) Allow() bool { r.mutex.Lock() defer r.mutex.Unlock() r.refillTokens() if r.tokens >= 1 { r.tokens-- return true } return false }
- 快速发送 3 个请求:全部都应该成功。
- 在同一分钟内发送第四个请求:您应该看到“超出速率限制”消息。
- 等待 20 秒,然后重试:存储桶重新装满,请求应该会成功。
源代码
GitHub 存储库
以上是初学者的速率限制:它是什么以及如何在 Go 中构建速率限制的详细内容。更多信息请关注PHP中文网其他相关文章!

Golangisidealforbuildingscalablesystemsduetoitsefficiencyandconcurrency,whilePythonexcelsinquickscriptinganddataanalysisduetoitssimplicityandvastecosystem.Golang'sdesignencouragesclean,readablecodeanditsgoroutinesenableefficientconcurrentoperations,t

Golang在并发性上优于C ,而C 在原始速度上优于Golang。1)Golang通过goroutine和channel实现高效并发,适合处理大量并发任务。2)C 通过编译器优化和标准库,提供接近硬件的高性能,适合需要极致优化的应用。

选择Golang的原因包括:1)高并发性能,2)静态类型系统,3)垃圾回收机制,4)丰富的标准库和生态系统,这些特性使其成为开发高效、可靠软件的理想选择。

Golang适合快速开发和并发场景,C 适用于需要极致性能和低级控制的场景。1)Golang通过垃圾回收和并发机制提升性能,适合高并发Web服务开发。2)C 通过手动内存管理和编译器优化达到极致性能,适用于嵌入式系统开发。

Golang在编译时间和并发处理上表现更好,而C 在运行速度和内存管理上更具优势。1.Golang编译速度快,适合快速开发。2.C 运行速度快,适合性能关键应用。3.Golang并发处理简单高效,适用于并发编程。4.C 手动内存管理提供更高性能,但增加开发复杂度。

Golang在Web服务和系统编程中的应用主要体现在其简洁、高效和并发性上。1)在Web服务中,Golang通过强大的HTTP库和并发处理能力,支持创建高性能的Web应用和API。2)在系统编程中,Golang利用接近硬件的特性和对C语言的兼容性,适用于操作系统开发和嵌入式系统。

Golang和C 在性能对比中各有优劣:1.Golang适合高并发和快速开发,但垃圾回收可能影响性能;2.C 提供更高性能和硬件控制,但开发复杂度高。选择时需综合考虑项目需求和团队技能。

Golang适合高性能和并发编程场景,Python适合快速开发和数据处理。 1.Golang强调简洁和高效,适用于后端服务和微服务。 2.Python以简洁语法和丰富库着称,适用于数据科学和机器学习。


热AI工具

Undresser.AI Undress
人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover
用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool
免费脱衣服图片

Clothoff.io
AI脱衣机

Video Face Swap
使用我们完全免费的人工智能换脸工具轻松在任何视频中换脸!

热门文章

热工具

Dreamweaver Mac版
视觉化网页开发工具

SublimeText3 Linux新版
SublimeText3 Linux最新版

SecLists
SecLists是最终安全测试人员的伙伴。它是一个包含各种类型列表的集合,这些列表在安全评估过程中经常使用,都在一个地方。SecLists通过方便地提供安全测试人员可能需要的所有列表,帮助提高安全测试的效率和生产力。列表类型包括用户名、密码、URL、模糊测试有效载荷、敏感数据模式、Web shell等等。测试人员只需将此存储库拉到新的测试机上,他就可以访问到所需的每种类型的列表。

SublimeText3 Mac版
神级代码编辑软件(SublimeText3)

SublimeText3汉化版
中文版,非常好用