0

0

Golang如何使用sync.Cond实现条件等待_Golang条件等待并发控制实践详解

P粉602998670

P粉602998670

发布时间:2025-11-07 10:48:03

|

772人浏览过

|

来源于php中文网

原创

sync.Cond用于协程间条件同步,需配合互斥锁使用,通过Wait()等待条件、Signal()/Broadcast()通知,应循环检查条件以防虚假唤醒。

golang如何使用sync.cond实现条件等待_golang条件等待并发控制实践详解

在Go语言中,sync.Cond 是一种用于协程间同步的机制,适用于某个条件未满足时让协程等待,直到其他协程通知条件已达成。它不是用来替代互斥锁或通道的,而是在特定场景下对并发控制的有力补充。本文将详细说明如何正确使用 sync.Cond 实现条件等待,并结合实际例子帮助理解其工作原理和最佳实践。

理解 sync.Cond 的基本结构

sync.Cond 由三部分组成:一个互斥锁(通常是 *sync.Mutex*sync.RWMutex)、一个条件变量和一组等待操作。它的核心方法包括:

  • Wait():释放锁并进入等待状态,直到被 Signal 或 Broadcast 唤醒,唤醒后重新获取锁。
  • Signal():唤醒一个正在等待的协程。
  • Broadcast():唤醒所有等待的协程。

必须注意的是,Wait() 调用前必须持有与 Cond 关联的锁,否则会引发 panic。

正确初始化与使用 Cond

创建 sync.Cond 时,应使用 sync.NewCond 并传入一个已初始化的锁。下面是一个标准的初始化方式:

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

mu := &sync.Mutex{}
cond := sync.NewCond(mu)

之后,在需要等待某个条件成立的地方使用 Wait(),例如等待缓冲区非空:

cond.L.Lock()
for len(buffer) == 0 {
    cond.Wait()
}
// 处理数据
data := buffer[0]
buffer = buffer[1:]
cond.L.Unlock()

这里使用 for 循环检查条件 而不是 if,是因为可能存在虚假唤醒(spurious wakeups),即协程被唤醒但条件仍未满足。因此必须循环检查条件是否真正成立。

viable
viable

基于GPT-4的AI非结构化数据分析平台

下载

通知等待方:Signal 与 Broadcast 的选择

当修改了共享状态并可能使等待条件成立时,需调用 Signal()Broadcast() 来唤醒等待者。

  • 使用 Signal() 当只有一个等待者需要被唤醒,比如生产者-消费者模型中放入一个元素,只需唤醒一个消费者。
  • 使用 Broadcast() 当多个等待者可能同时满足条件,例如关闭资源池时通知所有等待协程退出。

示例:生产者在向缓冲区添加数据后发出信号:

cond.L.Lock()
buffer = append(buffer, newData)
cond.L.Unlock()
cond.Signal() // 唤醒一个消费者

实战示例:线程安全的事件等待器

设想一个场景:主线程等待某个异步任务完成后再继续执行。可以使用 sync.Cond 实现事件驱动的等待逻辑。

package main

import ( "sync" "time" "fmt" )

func main() { mu := &sync.Mutex{} cond := sync.NewCond(mu) ready := false

// 模拟异步任务
go func() {
    time.Sleep(2 * time.Second)
    cond.L.Lock()
    ready = true
    cond.L.Unlock()
    cond.Broadcast() // 通知所有等待者
}()

// 主线程等待
cond.L.Lock()
for !ready {
    cond.Wait()
}
cond.L.Unlock()

fmt.Println("任务已完成,继续执行...")

}

这个例子展示了如何用 sync.Cond 实现“等待某一状态变为 true”的典型模式。注意,共享变量 ready 的读写都必须在锁保护下进行。

基本上就这些。合理使用 sync.Cond 可以写出高效且清晰的条件同步逻辑,关键在于始终配合锁使用、用 for 检查条件、正确选择通知方式。虽然 Go 更推荐使用 channel 进行协程通信,但在某些性能敏感或状态驱动的场景中,sync.Cond 依然是不可替代的工具

相关专题

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

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

173

2024.02.23

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

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

224

2024.02.23

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

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

334

2024.02.23

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

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

204

2024.03.05

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

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

387

2024.05.21

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

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

193

2025.06.09

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

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

184

2025.06.10

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

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

191

2025.06.17

苹果官网入口直接访问
苹果官网入口直接访问

苹果官网直接访问入口是https://www.apple.com/cn/,该页面具备0.8秒首屏渲染、HTTP/3与Brotli加速、WebP+AVIF双格式图片、免登录浏览全参数等特性。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

10

2025.12.24

热门下载

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

精品课程

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

共32课时 | 2.9万人学习

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

共10课时 | 0.8万人学习

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

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