登录
首页 >  Golang >  Go教程

Go语言如何用cond条件变量_Go语言sync.Cond教程【实用】

时间:2026-05-24 21:57:26 182浏览 收藏

Golang小白一枚,正在不断学习积累知识,现将学习到的知识记录一下,也是将我的所得分享给大家!而今天这篇文章《Go语言如何用cond条件变量_Go语言sync.Cond教程【实用】》带大家来了解一下##content_title##,希望对大家的知识积累有所帮助,从而弥补自己的不足,助力实战开发!


sync.Cond 不是 channel 替代品,须在 for 循环中调用 Wait 以防虚假唤醒;Signal/Broadcast 必须在锁内调用;Cond 必须绑定已初始化的 Mutex/RWMutex;Wait 返回时已持锁,需手动 Unlock。

Go语言如何用cond条件变量_Go语言sync.Cond教程【实用】

别用 sync.Cond 做一次性通知,也别裸写 cond.Wait() —— 它不是 channel 的替代品,而是为“多 goroutine 等同一动态条件”设计的协作原语。

Wait 必须在 for 循环里调用,不能用 if

虚假唤醒(spurious wakeup)是真实存在的:即使没人调用 SignalBroadcastWait() 也可能返回。此时若只用 if 判断一次,goroutine 就会误以为条件已满足,接着访问空队列、读取未初始化数据,或触发 panic。

  • 错误写法:if len(queue) == 0 { cond.Wait() } → 唤醒后不重检,风险极高
  • 正确写法:for len(queue) == 0 { cond.Wait() } → 每次唤醒都重新校验
  • 条件变量本身不保存状态,它只负责“挂起/唤醒”,真假全靠你用锁保护的变量来定义

Signal 和 Broadcast 必须在锁内调用才安全

虽然文档说 SignalBroadcast 不强制要求持锁,但实际中必须放在 Lock() / Unlock() 之间。否则可能出现“通知丢失”:等待者刚进入 Wait() 的释放锁阶段,通知就发了,结果没人收到。

  • 典型顺序:cond.L.Lock(); updateSharedState(); cond.Signal(); cond.L.Unlock()
  • 不要把 Signal 放在 Unlock() 之后 —— 中间可能有其他 goroutine 抢入并修改状态
  • Broadcast 在上百 goroutine 等待时会引发惊群(thundering herd),唤醒全部再抢锁,多数又立刻 Wait 回去;高并发下优先考虑分批唤醒或换 channel + select

Cond 必须和 sync.Mutex 或 sync.RWMutex 绑定,不能单独 new

sync.Cond 本身不带锁,它的 L 字段必须指向一个已初始化的互斥锁实例。漏掉这步,Wait() 会 panic,或者导致竞态——因为条件变量的读写根本没被保护。

  • 合法初始化:mu := &sync.Mutex{}; cond := sync.NewCond(mu)
  • 非法操作:cond := sync.NewCond(&sync.Mutex{}) → 锁是临时对象,地址可能失效
  • 条件变量和锁是一对一绑定关系,多个 Cond 共享同一个锁是允许的,但每个 Cond 必须有自己的 L 指向
  • 别用 sync.RWMutexRLock() 配合 Wait() —— Wait() 内部调用的是 L.Unlock(),而 RLock() 对应的是 RUnlock(),类型不匹配会 panic

最常被忽略的一点:每次 Wait() 返回时,你已经重新持有锁了,但很多人忘了在逻辑末尾 Unlock(),结果整个临界区被锁死。这不是 Cond 的 bug,是你没遵守“加锁 → 检查 → 等待 → 操作 → 解锁”的闭环。

今天关于《Go语言如何用cond条件变量_Go语言sync.Cond教程【实用】》的内容介绍就到此结束,如果有什么疑问或者建议,可以在golang学习网公众号下多多回复交流;文中若有不正之处,也希望回复留言以告知!

资料下载
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>