Go语言中的信号量:原理与实践指南
Go语言中的信号量:原理与实践指南
引言
在并发编程中,控制对共享资源的访问是一个经典问题。Go语言提供了丰富的并发原语(如sync.Mutex),但当我们需要灵活限制并发数量时,信号量(Semaphore)便成为重要工具。本文将深入解析Go中信号量的实现方式,并通过代码示例演示其典型应用场景。
一、信号量基础
什么是信号量?
信号量是一种同步机制,用于限制同时访问某资源的线程(或goroutine)数量。其核心是一个计数器,操作包括:
- P操作(获取):计数器减1,若计数器为0则阻塞等待
- V操作(释放):计数器加1,唤醒等待的线程
与互斥锁(Mutex)的区别:
| 特性 | 互斥锁 | 信号量 |
|---|---|---|
| 并发限制数量 | 1 | 可自定义(N≥1) |
| 适用场景 | 严格互斥访问 | 流量控制、资源池 |
二、Go中的两种实现方案
方案1:基于Channel的实现(标准库方式)
go
package mainimport (
"fmt"
"sync"
"time"
)func main() {
const maxConcurrent = 2 // 最大并发数
sem := make(chan struct{}, maxConcurrent)
var wg sync.WaitGroupfor i := 1; i <= 5; i++ {wg.Add(1)go func(id int) {defer wg.Done()sem <- struct{}{} // 获取信号量defer func() { <-sem }() // 释放信号量fmt.Printf("Worker %d started\n", id)time.Sleep(time.Second) // 模拟工作负载fmt.Printf("Worker %d done\n", id)}(i)}wg.Wait()fmt.Println("All workers completed")
}
代码解析:
sem := make(chan struct{}, N)创建容量为N的缓冲通道sem <- struct{}{}通过发送空结构体占用槽位<-sem接收数据释放槽位defer确保无论流程如何都会释放资源
方案2:使用semaphore.Weighted(扩展库实现)
bash
go get golang.org/x/sync/semaphore # 安装依赖
go
package mainimport (
"context"
"fmt"
"golang.org/x/sync/semaphore"
"sync"
"time"
)func main() {
const (
maxConcurrent = 2 // 最大并发数
totalWorkers = 5 // 总任务数
)sem := semaphore.NewWeighted(maxConcurrent)ctx := context.Background()var wg sync.WaitGroupfor i := 1; i <= totalWorkers; i++ {wg.Add(1)go func(id int) {defer wg.Done()// 尝试获取信号量if err := sem.Acquire(ctx, 1); err != nil {fmt.Printf("Worker %d failed: %v\n", id, err)return}defer sem.Release(1)fmt.Printf("Worker %d started\n", id)time.Sleep(time.Second)fmt.Printf("Worker %d done\n", id)}(i)}wg.Wait()fmt.Println("All workers completed")
}
特性说明:
- 支持加权请求(如一次申请多个许可)
- 可结合
context.Context实现超时控制 - 更适用于复杂资源管理场景
三、关键应用场景
1. 数据库连接池控制
go
// 创建最大10连接的信号量
var dbSem = semaphore.NewWeighted(10)func QueryDatabase(query string) {
dbSem.Acquire(context.Background(), 1)
defer dbSem.Release(1)// 执行数据库操作
}
2. 限流下载器
go
// 限制同时下载数为3
var downloadSem = make(chan struct{}, 3)func DownloadFile(url string) {
downloadSem <- struct{}{}
defer func() { <-downloadSem }()// 执行下载逻辑
}
3. 批量任务分流
go
// 控制100个并发处理任务
sem := semaphore.NewWeighted(100)
for _, task := range tasks {
go func(t Task) {
sem.Acquire(ctx, 1)
defer sem.Release(1)
process(t)
}(task)
}
四、实现方案对比
| 维度 | Channel实现 | semaphore.Weighted |
|---|---|---|
| 标准库支持 | ✅ 无需额外依赖 | ❌ 需要安装扩展库 |
| 加权请求 | ❌ 不支持 | ✅ 支持 |
| 超时控制 | 需搭配select实现 | ✅ 原生支持Context |
| 易用性 | 简单场景推荐 | 复杂场景推荐 |
| 性能开销 | 较低 | 略高(含锁机制) |
五、最佳实践建议
-
资源释放
始终使用defer释放信号量,避免协程异常导致资源泄漏 -
容量规划
根据实际硬件资源(CPU核心数、IO带宽等)设置合理并发数 -
异常处理
使用semaphore.Weighted时检查Acquire()返回的error -
调试技巧
添加指标监控当前信号量使用率:
go
fmt.Printf(“Available: %d/%d\n”, len(sem), cap(sem))
结语
信号量为Go并发编程提供了灵活的资源管控能力。无论是简单的通道实现,还是功能更强的semaphore.Weighted,开发者都可以根据具体需求选择合适的方案。合理使用信号量不仅能提升程序稳定性,还能有效避免资源竞争导致的性能瓶颈。
扩展阅读:
- Go官方并发指南
semaphore包源码分析
相关文章:
Go语言中的信号量:原理与实践指南
Go语言中的信号量:原理与实践指南 引言 在并发编程中,控制对共享资源的访问是一个经典问题。Go语言提供了丰富的并发原语(如sync.Mutex),但当我们需要灵活限制并发数量时,信号量(Semaphore&am…...
Qt如何将数据传入labview,Qt又如何从labview中读取数据?
Qt如何将数据传入labview,Qt又如何从labview中读取数据? Qt如何将数据传入labviewQt如何从labview中读取数据 Qt如何将数据传入labview Qt如何从labview中读取数据...
vue3学习2
ts定义接口: 引入的时候要加type: 调用: ts创建自定义type类型,引入的时候也要加type: reactive可以直接传泛型: 加?声明不强制: defineProps接收父组件传递的props,其中defineProp…...
spring中手写注解开发(创建对象篇)
说明: 在spring底层中并不是我写的如此,这篇只是我用我自己的方式实现了使用注解组件扫描并且 创建对象,方法并不是很难,可以看一看,欢迎大佬评论 第一步: 我们需要自己写一个注解,我用的是idea…...
Android OpenGLES2.0开发(十一):渲染YUV
人生如逆旅,我亦是行人 Android OpenGLES开发:EGL环境搭建Android OpenGLES2.0开发(一):艰难的开始Android OpenGLES2.0开发(二):环境搭建Android OpenGLES2.0开发(三&am…...
在linux中利用conda安装blast
在 Linux 中使用 conda 安装 BLAST 非常简单。conda 是一个流行的包管理工具,可以轻松安装和管理生物信息学工具,包括 BLAST。以下是具体步骤: 1. 确保已安装 Conda 如果你还没有安装 conda,可以参考以下步骤安装 Miniconda&…...
三、多项式环
文章目录 一、多项式环的定义二、多项式环的性质1. 多项式加法2. 多项式乘法3. 满足的运算规律4. 次数5. 单位元 三、剩余多项式环(商多项式环)四、有限多项式环五、多项式环的性质与特性1. 子环与理想2. 不可约性和素性3. 有限生成性 一、多项式环的定义…...
python unzip file
要在 Python 中解压文件并显示进度,我们需要在解压过程中跟踪文件的提取进度。由于 zipfile 模块本身不直接支持进度显示,我们可以通过手动计算并使用 tqdm 库来显示进度条。 安装 tqdm 首先,确保你已经安装了 tqdm 库,用于显示…...
MySQL-增删改查
一、Create(创建) 📖 语法: INSERT INTO table_name(value_list); 当我们使用表的时候,就可以使用这个语法来向表中插入元素~ 我们这边创建一个用于示范的表(Student)~ create table student( id int, name varchar(20), chinese int, math…...
LeetCode 热题100 15. 三数之和
LeetCode 热题100 | 15. 三数之和 大家好,今天我们来解决一道经典的算法题——三数之和。这道题在 LeetCode 上被标记为中等难度,要求我们从一个整数数组中找到所有不重复的三元组,使得三元组的和为 0。下面我将详细讲解解题思路,…...
网络空间安全(1)web应用程序的发展历程
前言 Web应用程序的发展历程是一部技术创新与社会变革交织的长卷,从简单的文档共享系统到如今复杂、交互式、数据驱动的平台,经历了多个重要阶段。 一、起源与初期发展(1989-1995年) Web的诞生: 1989年,欧洲…...
ABAQUS功能梯度材料FGM模型
功能梯度材料(FGM)作为一种新型复合材料,通过材料内部成分或微观结构的梯度变化,优化特定性能适应复杂环境,被广泛应用于高温防护、结构优化、生物医学、光电设备等领域。本案例介绍在ABAQUS内建立功能梯度材料模型。 …...
自适应增强技术
1. 传统图像处理中的自适应增强(如CLAHE) 难度:⭐容易 实现方式:调用成熟的库(如OpenCV)函数即可完成。 示例代码(CLAHE增强): <PYTHON> import cv2# 输入灰度或彩…...
虚拟项目:一个好用的工具平台
在当今数字化的时代,虚拟项目如雨后春笋般涌现,为人们提供了诸多便捷且充满机遇的选择。以下将为大家详细介绍几种颇具特色的虚拟项目,包括书签、资源站、题库、虚拟商城、专栏、证件照以及分站搭建等,一起来了解它们各自的独特之…...
MySQL 和 Elasticsearch 之间的数据同步
MySQL 和 Elasticsearch 之间的数据同步是常见的需求,通常用于将结构化数据从关系型数据库同步到 Elasticsearch 以实现高效的全文搜索、聚合分析和实时查询。以下是几种常用的同步方案及其实现方法: 1. 应用层双写(双写模式) 原…...
PS裁剪工具
裁剪: 多张图同一标准裁剪:裁剪–》前面的图像–》选择其他图像–》 确定 选区–》裁剪工具–》确定:选区制作矩形裁剪 裁剪–》拉直 裁剪–》内容识别:当裁剪大于图片大小,会自动填充空白区域 (栅格化图层…...
[Web 安全] PHP 反序列化漏洞 —— PHP 序列化 反序列化
关注这个专栏的其他相关笔记:[Web 安全] 反序列化漏洞 - 学习笔记-CSDN博客 0x01:PHP 序列化 — Serialize 序列化就是将对象的状态信息转化为可以存储或传输的形式的过程,在 PHP 中,通常使用 serialize() 函数来完成序列化的操作…...
QT入门--QMainWindow
从上向下依次是菜单栏,工具栏,铆接部件(浮动窗口),状态栏,中心部件 菜单栏 创建菜单栏 QMenuBar* mybar1 menuBar(); 将菜单栏放到窗口中 setMenuBar(mybar1); 创建菜单 QMenu *myfilemenu mybar1-…...
C++ | 高级教程 | 信号处理
👻 概念 信号 —— 操作系统传给进程的中断,会提早终止程序有些信号不能被程序捕获,有些则可以被捕获,并基于信号采取适当的动作 信号描述SIGABRT程序的异常终止,如调用 abortSIGFPE错误的算术运算,比如除…...
最新前端框架选型对比与建议(React/Vue/Svelte/Angular)
前端框架选型对比与建议(React/Vue/Svelte/Angular) 一、核心框架技术特性对比(基于最新版本) 维度React 19 25Vue 3.5 12Svelte 5 25Angular 19 5核心理念函数式编程、JSX语法、虚拟DOM渐进式框架、组合式API、模板语法编译时框…...
Prompt Tuning、P-Tuning、Prefix Tuning的区别
一、Prompt Tuning、P-Tuning、Prefix Tuning的区别 1. Prompt Tuning(提示调优) 核心思想:固定预训练模型参数,仅学习额外的连续提示向量(通常是嵌入层的一部分)。实现方式:在输入文本前添加可训练的连续向量(软提示),模型只更新这些提示参数。优势:参数量少(仅提…...
DockerHub与私有镜像仓库在容器化中的应用与管理
哈喽,大家好,我是左手python! Docker Hub的应用与管理 Docker Hub的基本概念与使用方法 Docker Hub是Docker官方提供的一个公共镜像仓库,用户可以在其中找到各种操作系统、软件和应用的镜像。开发者可以通过Docker Hub轻松获取所…...
Admin.Net中的消息通信SignalR解释
定义集线器接口 IOnlineUserHub public interface IOnlineUserHub {/// 在线用户列表Task OnlineUserList(OnlineUserList context);/// 强制下线Task ForceOffline(object context);/// 发布站内消息Task PublicNotice(SysNotice context);/// 接收消息Task ReceiveMessage(…...
自然语言处理——Transformer
自然语言处理——Transformer 自注意力机制多头注意力机制Transformer 虽然循环神经网络可以对具有序列特性的数据非常有效,它能挖掘数据中的时序信息以及语义信息,但是它有一个很大的缺陷——很难并行化。 我们可以考虑用CNN来替代RNN,但是…...
代理篇12|深入理解 Vite中的Proxy接口代理配置
在前端开发中,常常会遇到 跨域请求接口 的情况。为了解决这个问题,Vite 和 Webpack 都提供了 proxy 代理功能,用于将本地开发请求转发到后端服务器。 什么是代理(proxy)? 代理是在开发过程中,前端项目通过开发服务器,将指定的请求“转发”到真实的后端服务器,从而绕…...
关键领域软件测试的突围之路:如何破解安全与效率的平衡难题
在数字化浪潮席卷全球的今天,软件系统已成为国家关键领域的核心战斗力。不同于普通商业软件,这些承载着国家安全使命的软件系统面临着前所未有的质量挑战——如何在确保绝对安全的前提下,实现高效测试与快速迭代?这一命题正考验着…...
云原生玩法三问:构建自定义开发环境
云原生玩法三问:构建自定义开发环境 引言 临时运维一个古董项目,无文档,无环境,无交接人,俗称三无。 运行设备的环境老,本地环境版本高,ssh不过去。正好最近对 腾讯出品的云原生 cnb 感兴趣&…...
sipsak:SIP瑞士军刀!全参数详细教程!Kali Linux教程!
简介 sipsak 是一个面向会话初始协议 (SIP) 应用程序开发人员和管理员的小型命令行工具。它可以用于对 SIP 应用程序和设备进行一些简单的测试。 sipsak 是一款 SIP 压力和诊断实用程序。它通过 sip-uri 向服务器发送 SIP 请求,并检查收到的响应。它以以下模式之一…...
三分算法与DeepSeek辅助证明是单峰函数
前置 单峰函数有唯一的最大值,最大值左侧的数值严格单调递增,最大值右侧的数值严格单调递减。 单谷函数有唯一的最小值,最小值左侧的数值严格单调递减,最小值右侧的数值严格单调递增。 三分的本质 三分和二分一样都是通过不断缩…...
GO协程(Goroutine)问题总结
在使用Go语言来编写代码时,遇到的一些问题总结一下 [参考文档]:https://www.topgoer.com/%E5%B9%B6%E5%8F%91%E7%BC%96%E7%A8%8B/goroutine.html 1. main()函数默认的Goroutine 场景再现: 今天在看到这个教程的时候,在自己的电…...
