0

0

Golang HTTP请求限流与并发控制项目

P粉602998670

P粉602998670

发布时间:2025-10-10 17:00:01

|

444人浏览过

|

来源于php中文网

原创

使用令牌桶和信号量机制控制限流与并发,保障Golang HTTP服务稳定性。首先通过golang.org/x/time/rate实现每秒10个请求、突发50的令牌桶限流;接着用带缓冲channel(容量5)限制最大并发数,防止资源耗尽;最后结合IP级限流管理器,按IP维度分配独立令牌桶,并定期清理长时间未活跃的IP记录,实现精细化控制。实际应用中建议结合监控动态调整阈值。

golang http请求限流与并发控制项目

在构建高并发的 Golang HTTP 服务时,控制请求频率和并发量是保障系统稳定的关键。不加限制的请求可能导致后端资源耗尽、响应延迟上升甚至服务崩溃。通过合理的限流与并发控制机制,可以有效保护服务稳定性,提升用户体验。

使用令牌桶实现HTTP请求限流

令牌桶算法是一种常见且灵活的限流方式,它允许一定程度的突发流量,同时保证平均速率不超过设定值。Golang 标准库 golang.org/x/time/rate 提供了开箱即用的令牌桶实现。

以下是一个基于 rate.Limiter 的中间件示例:

package main

import ( "golang.org/x/time/rate" "net/http" "time" )

var limiter = rate.NewLimiter(10, 50) // 每秒10个令牌,最多容纳50个

立即学习go语言免费学习笔记(深入)”;

func rateLimit(next http.HandlerFunc) http.HandlerFunc { return func(w http.ResponseWriter, r *http.Request) { if !limiter.Allow() { http.StatusText(http.StatusTooManyRequests) http.Error(w, "请求过于频繁", http.StatusTooManyRequests) return } next.ServeHTTP(w, r) } }

func handler(w http.ResponseWriter, r *http.Request) { w.Write([]byte("处理请求")) }

func main() { mux := http.NewServeMux() mux.HandleFunc("/", rateLimit(handler)) http.ListenAndServe(":8080", mux) }

上面代码中,每秒最多处理10个请求,最多可积压40个(burst=50)。适合用于API接口防刷或防止爬虫滥用。

控制最大并发请求

除了按时间频率限流,有时需要硬性限制同时处理的请求数量,防止资源(如数据库连接、内存)被耗尽。可以通过带缓冲的 channel 实现信号量机制。

示例:限制最多同时处理5个请求

LongShot
LongShot

LongShot 是一款 AI 写作助手,可帮助您生成针对搜索引擎优化的内容博客。

下载
var sem = make(chan struct{}, 5)

func concurrencyLimit(next http.HandlerFunc) http.HandlerFunc { return func(w http.ResponseWriter, r *http.Request) { select { case sem <- struct{}{}: // 获取执行权 default: http.Error(w, "服务繁忙,请稍后再试", http.StatusServiceUnavailable) return } defer func() { <-sem }() // 释放 next.ServeHTTP(w, r) } }

该方法简单有效,适用于 IO 密集型任务较多、资源敏感的服务场景。

结合限流与并发控制的完整结构

实际项目中,通常将两种策略结合使用。例如:先通过并发控制防止系统过载,再用限流器平滑请求分布。

可以封装一个通用的限流管理器,支持按 IP 或用户维度进行控制:

type IpLimiter struct {
    mu sync.RWMutex
    limiters map[string]*rate.Limiter
    lastSeen map[string]time.Time
    r  float64
    b int
}

func NewIpLimiter(r float64, b int) IpLimiter { il := &IpLimiter{ limiters: make(map[string]rate.Limiter), lastSeen: make(map[string]time.Time), r: r, b: b, } go il.cleanup() return il }

func (il IpLimiter) GetLimiter(ip string) rate.Limiter { il.mu.Lock() defer il.mu.Unlock()

limiter, exists := il.limiters[ip]
if !exists {
    limiter = rate.NewLimiter(il.r, il.b)
    il.limiters[ip] = limiter
    il.lastSeen[ip] = time.Now()
} else {
    il.lastSeen[ip] = time.Now()
}
return limiter

}

func (il IpLimiter) cleanup() { for { time.Sleep(time.Minute) il.mu.Lock() for ip, last := range il.lastSeen { if time.Since(last) > 3time.Minute { delete(il.limiters, ip) delete(il.lastSeen, ip) } } il.mu.Unlock() } }

在中间件中调用:

var ipLimiter = NewIpLimiter(1, 5) // 每秒1次,最多5次突发

func limitByIP(next http.HandlerFunc) http.HandlerFunc { return func(w http.ResponseWriter, r *http.Request) { ip := r.RemoteAddr if !ipLimiter.GetLimiter(ip).Allow() { http.Error(w, "访问过于频繁", http.StatusTooManyRequests) return } next(w, r) } }

基本上就这些。通过合理组合令牌桶限流与 channel 控制并发,可以构建出稳定可靠的 HTTP 服务。关键是根据业务场景选择合适的阈值,并配合监控及时调整策略。

相关专题

更多
golang如何定义变量
golang如何定义变量

golang定义变量的方法:1、声明变量并赋予初始值“var age int =值”;2、声明变量但不赋初始值“var age int”;3、使用短变量声明“age :=值”等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

174

2024.02.23

golang有哪些数据转换方法
golang有哪些数据转换方法

golang数据转换方法:1、类型转换操作符;2、类型断言;3、字符串和数字之间的转换;4、JSON序列化和反序列化;5、使用标准库进行数据转换;6、使用第三方库进行数据转换;7、自定义数据转换函数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

225

2024.02.23

golang常用库有哪些
golang常用库有哪些

golang常用库有:1、标准库;2、字符串处理库;3、网络库;4、加密库;5、压缩库;6、xml和json解析库;7、日期和时间库;8、数据库操作库;9、文件操作库;10、图像处理库。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

335

2024.02.23

golang和python的区别是什么
golang和python的区别是什么

golang和python的区别是:1、golang是一种编译型语言,而python是一种解释型语言;2、golang天生支持并发编程,而python对并发与并行的支持相对较弱等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

206

2024.03.05

golang是免费的吗
golang是免费的吗

golang是免费的。golang是google开发的一种静态强类型、编译型、并发型,并具有垃圾回收功能的开源编程语言,采用bsd开源协议。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

388

2024.05.21

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

194

2025.06.09

golang相关判断方法
golang相关判断方法

本专题整合了golang相关判断方法,想了解更详细的相关内容,请阅读下面的文章。

189

2025.06.10

golang数组使用方法
golang数组使用方法

本专题整合了golang数组用法,想了解更多的相关内容,请阅读专题下面的文章。

191

2025.06.17

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

74

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Go 教程
Go 教程

共32课时 | 3.2万人学习

Go语言实战之 GraphQL
Go语言实战之 GraphQL

共10课时 | 0.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号