0

0

Go并发编程什么时候用锁_Go锁与channel选择建议

P粉602998670

P粉602998670

发布时间:2026-01-15 11:13:08

|

284人浏览过

|

来源于php中文网

原创

必须用 sync.Mutex 或 sync.RWMutex 保护共享内存的非原子读写;优先用 atomic 处理单个整数/指针的简单操作;channel 适用于通信协作而非替代锁;复杂状态需锁、channel、WaitGroup、atomic 组合使用。

go并发编程什么时候用锁_go锁与channel选择建议

共享内存修改必须用 sync.Mutexsync.RWMutex

当多个 goroutine 同时读写同一块内存(比如一个全局 map、计数器 int、或结构体字段),且写操作不是原子的,就必须加锁。Go 的 map 本身不支持并发写,直接写会 panic:fatal error: concurrent map writes;即使只是“读+写”组合(如 counter++),也非原子,必须保护。

实操建议:

  • sync.Mutex 适用于读写都较频繁、或写操作占比较高的场景
  • sync.RWMutex 更适合读多写少(比如配置缓存、状态快照),允许多个 goroutine 同时读,但写时会独占
  • 锁粒度要细:不要对整个函数加锁,优先锁具体字段或局部数据结构;避免把 http.Handler 整个方法体包在 mu.Lock()
  • 务必检查锁的生命周期:不要在 defer 中 unlock 一个未 lock 的 mutex,也不要跨 goroutine 传递已 lock 的 mutex

channel 更适合 goroutine 间协作与流程控制

channel 不是“替代锁的通用方案”,而是为通信而生。它天然适合表达“谁等谁”“谁通知谁”“谁提供数据谁消费数据”这类关系。比如 worker pool、任务分发、信号通知(done channel)、超时控制(select + time.After)。

常见误用:

  • chan struct{} 当作“锁开关”来保护临界区——这本质是用 channel 模拟锁,但语义不清、易死锁、性能更差
  • 为每个简单计数器开一个 chan int,再起 goroutine 去收——过度工程,远不如 atomic.AddInt64mu.Lock() 直观高效
  • 在 hot path(如每毫秒调用一次的统计函数)中使用 unbuffered channel —— 会强制 goroutine 切换和调度开销,比锁还重

能用 atomic 就别用锁或 channel

对单个整数、指针、unsafe.Pointer 的读写,sync/atomic 提供零分配、无锁、高性能的原子操作。它比 mutex 轻量得多,也比 channel 更直接。

Insou AI
Insou AI

Insou AI 是一款强大的人工智能助手,旨在帮助你轻松创建引人入胜的内容和令人印象深刻的演示。

下载

适用条件明确:

  • 操作目标是 int32int64uint32uint64uintptrunsafe.Pointer 或其指针
  • 只需要基础操作:增减(atomic.AddInt64)、比较并交换(atomic.CompareAndSwapInt64)、加载(atomic.LoadInt64)、存储(atomic.StoreInt64
  • 不需要复合逻辑(比如“如果值小于 100 才加 1”且该判断+写入需原子)——这时得回退到 mutex
var counter int64
// 安全:原子增
atomic.AddInt64(&counter, 1)

// 危险:非原子,竞态
counter++

复杂状态同步往往需要锁 + channel 组合

纯 channel 很难表达“带状态的响应式行为”。例如实现一个带缓冲、可关闭、支持等待所有任务完成的 job queue,通常要:

  • 用 channel 接收新任务(通信)
  • sync.WaitGroup 记录活跃 worker(协作生命周期)
  • sync.RWMutex 保护内部状态(如当前积压任务数、是否已关闭)
  • atomic.Bool 标记关闭信号(轻量状态)

单独依赖某一种机制,容易在边界情况(如关闭中又投递任务、worker panic 后状态残留)下出错。真正健壮的并发组件,往往是多种原语按职责混合使用。

最容易被忽略的是:锁和 channel 的错误处理路径是否一致。比如在 select 中从 channel 收消息后 panic 了,mutex 是否仍处于 locked 状态?这类问题不会报编译错误,但会导致后续 goroutine 永久阻塞。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
scripterror怎么解决
scripterror怎么解决

scripterror的解决办法有检查语法、文件路径、检查网络连接、浏览器兼容性、使用try-catch语句、使用开发者工具进行调试、更新浏览器和JavaScript库或寻求专业帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

492

2023.10.18

500error怎么解决
500error怎么解决

500error的解决办法有检查服务器日志、检查代码、检查服务器配置、更新软件版本、重新启动服务、调试代码和寻求帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

382

2023.10.25

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

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

490

2025.06.09

golang结构体方法
golang结构体方法

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

202

2025.07.04

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

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

490

2025.06.09

golang结构体方法
golang结构体方法

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

202

2025.07.04

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

1051

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

615

2024.08.29

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

49

2026.03.13

热门下载

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

精品课程

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

共32课时 | 6.2万人学习

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

共10课时 | 0.9万人学习

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

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