sync.Cond用于协程间条件同步,需配合互斥锁使用,通过Wait()等待、Signal()/Broadcast()唤醒,确保条件判断与等待的原子性,避免忙等。

在Go语言中,sync.Cond 用于实现条件同步,它允许协程在某个条件满足时才继续执行。与互斥锁不同,sync.Cond 能够让协程等待某个特定条件成立,避免忙等,提高效率。它通常和 sync.Mutex 或 sync.RWMutex 配合使用。
sync.Cond 的基本结构
sync.Cond 包含三个核心方法:
Wait():释放锁并进入等待状态,直到被 Signal 或 Broadcast 唤醒。Signal():唤醒一个正在等待的协程。Broadcast():唤醒所有等待的协程。
Cond 必须配合互斥锁使用,因为条件判断和等待操作需要原子性保护。
如何创建和使用 sync.Cond
创建 Cond 实例时,需传入一个已初始化的锁(通常是 *sync.Mutex):
立即学习“go语言免费学习笔记(深入)”;
mu := &sync.Mutex{}cond := sync.NewCond(mu)
典型使用模式如下:
cond.L.Lock()for 条件不满足 { cond.Wait()}// 执行条件满足后的操作cond.L.Unlock()
注意:必须在循环中检查条件,防止虚假唤醒。
实际示例:生产者-消费者模型
以下是一个简单的生产者-消费者例子,使用 sync.Cond 控制对共享缓冲区的访问:
package mainimport ("fmt""sync""time")
var (buffer = make([]int, 0, 10)cond = sync.NewCond(&sync.Mutex{})isEmpty = true)
func producer() {for i := 0; i < 5; i++ {cond.L.Lock()buffer = append(buffer, i)isEmpty = falsefmt.Printf("生产者: 添加 %dn", i)cond.L.Unlock()cond.Broadcast() // 唤醒所有消费者time.Sleep(500 * time.Millisecond)}}
func consumer(name string) {for {cond.L.Lock()for isEmpty {cond.Wait() // 等待数据可用}item := buffer[0]buffer = buffer[1:]if len(buffer) == 0 {isEmpty = true}fmt.Printf("%s: 消费 %dn", name, item)cond.L.Unlock()time.Sleep(800 * time.Millisecond)}}
func main() {go producer()go consumer("消费者A")go consumer("消费者B")
time.Sleep(6 * time.Second)
}
在这个例子中,消费者在缓冲区为空时调用 Wait() 进入等待,生产者每次添加数据后调用 Broadcast() 通知所有消费者。Broadcast 能确保至少一个消费者被唤醒处理新数据。
Signal 和 Broadcast 的选择
如果只有一个协程在等待某个条件,使用 Signal() 更高效。如果有多个等待者,或不确定等待数量,应使用 Broadcast()。例如,在上述消费者模型中,多个消费者都在等待数据,使用 Broadcast 可以唤醒所有等待者,由它们竞争获取数据。
基本上就这些。sync.Cond 不复杂但容易忽略细节,关键是配合锁使用、在循环中检查条件,并合理选择唤醒方式。
以上就是Golang如何使用sync.Cond实现条件同步的详细内容,更多请关注创想鸟其它相关文章!
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1420921.html
微信扫一扫
支付宝扫一扫