Golang并发请求限制技巧分享
时间:2026-01-29 11:36:40 289浏览 收藏
积累知识,胜过积蓄金银!毕竟在Golang开发的过程中,会遇到各种各样的问题,往往都是一些细节知识点还没有掌握好而导致的,因此基础知识点的积累是很重要的。下面本文《Golang限制并发请求的常用方法》,就带大家讲解一下知识点,若是你对本文感兴趣,或者是想搞懂其中某个知识点,就请你继续往下看吧~
推荐用 golang.org/x/sync/semaphore 控制并发:NewWeighted 设置最大并发数,Acquire/Release 配对使用 defer 保证释放,支持超时与非阻塞 TryAcquire。

用 semaphore 控制并发请求数(推荐)
Go 标准库没有内置信号量,但可以用 sync.Mutex + sync.Cond 或第三方库(如 golang.org/x/sync/semaphore)实现。后者更轻量、语义清晰,是目前最主流的做法。
注意:不要自己手写带计数的互斥锁——容易漏掉 Unlock 或在 panic 时未释放,导致死锁或资源耗尽。
semaphore.Weighted支持非阻塞获取(TryAcquire),适合做快速失败判断- 每个请求应对应一次
Acquire和一次Release,建议用defer保证释放 - 初始权重值即最大并发数,比如
semaphore.NewWeighted(10)表示最多 10 个并发
import "golang.org/x/sync/semaphore"
<p>var sem = semaphore.NewWeighted(5)</p><p>func handleRequest() {
if err := sem.Acquire(context.Background(), 1); err != nil {
// 超时或上下文取消
return
}
defer sem.Release(1)</p><pre class="brush:php;toolbar:false;">// 执行实际请求逻辑
doHTTPCall()}
用 channel 做固定容量的并发令牌池
本质是用带缓冲的 channel 当作“令牌桶”,每次请求前从 channel 取一个空位,完成后放回。简单直接,不依赖外部包,适合轻量场景。
缺点是无法设置超时等待、不能动态调整容量,且容易误用:chan struct{} 容量设错或忘记 close 后的读取会 panic。
- 必须初始化为带缓冲的 channel:
make(chan struct{}, 10) - 获取令牌用
select+default可做非阻塞尝试 - 务必确保每次成功获取后都有对应的
send回 channel,否则池子会逐渐枯竭
var token = make(chan struct{}, 3)
<p>func init() {
for i := 0; i < cap(token); i++ {
token <- struct{}{}
}
}</p><p>func handleRequest() {
select {
case <-token:
// 获取成功
defer func() { token <- struct{}{} }()
doHTTPCall()
default:
// 无可用令牌,快速失败
return
}
}</p>用 worker pool 模式统一调度请求
当并发控制只是起点,后续还要做任务排队、优先级、重试、统计等,就该上 worker pool。它把“限制并发”变成“固定数量 goroutine 持续消费任务队列”,更易扩展和观测。
注意:别让 worker 数量远小于任务量又不加队列缓冲,会导致大量任务被丢弃;也别让队列无限增长,引发 OOM。
- worker 数量即最大并发数,通常设为 CPU 核心数或 IO 密集型场景下的经验值(如 10–50)
- 任务 channel 应设缓冲(如
make(chan *Request, 1000)),避免生产者阻塞 - 需显式关闭 channel 并等待所有 worker 退出,否则
range不会结束
func startWorkerPool(n int, jobs <h3>别踩 <code>time.Sleep</code> 或 <code>runtime.Gosched</code> 的坑</h3><p>有人试图用休眠或主动让出调度来“限速”,这是无效的。它们既不限制并发数,也不控制资源占用,只拖慢单个 goroutine,反而可能因堆积更多 goroutine 加剧内存压力。</p><p>尤其要注意:在 HTTP handler 中用 <code>time.Sleep</code> 等待,会占用整个 goroutine 直到超时,而 Go 的 HTTP server 默认为每个请求启一个 goroutine——等于人为制造 goroutine 泄漏。</p>
time.Sleep是时间控制,不是并发控制runtime.Gosched只建议用于防止长时间独占 P,不解决并发上限问题- 真正要控的是“同时执行的请求数”,不是“请求之间隔多久”
真正难的不是选哪种方式,而是确定那个数字:最大并发数设成多少?它得结合下游服务的吞吐能力、本机 CPU/内存余量、请求平均耗时和 p99 延迟目标来反复压测,而不是拍脑袋填个 10 或 100。
好了,本文到此结束,带大家了解了《Golang并发请求限制技巧分享》,希望本文对你有所帮助!关注golang学习网公众号,给大家分享更多Golang知识!
-
505 收藏
-
503 收藏
-
502 收藏
-
502 收藏
-
502 收藏
-
154 收藏
-
345 收藏
-
190 收藏
-
178 收藏
-
257 收藏
-
461 收藏
-
451 收藏
-
477 收藏
-
358 收藏
-
456 收藏
-
345 收藏
-
301 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习