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

在Go语言中,sync.Cond 用于实现条件同步,它允许协程在某个条件满足时才继续执行。与互斥锁不同,sync.Cond 能够让协程等待某个特定条件成立,避免忙等,提高效率。它通常和 sync.Mutex 或 sync.RWMutex 配合使用。
sync.Cond 包含三个核心方法:
Cond 必须配合互斥锁使用,因为条件判断和等待操作需要原子性保护。
创建 Cond 实例时,需传入一个已初始化的锁(通常是 *sync.Mutex):
立即学习“go语言免费学习笔记(深入)”;
mu := &sync.Mutex{}
cond := sync.NewCond(mu)
典型使用模式如下:
cond.L.Lock()
for 条件不满足 {
cond.Wait()
}
// 执行条件满足后的操作
cond.L.Unlock()
注意:必须在循环中检查条件,防止虚假唤醒。
以下是一个简单的生产者-消费者例子,使用 sync.Cond 控制对共享缓冲区的访问:
package main
<p>import (
"fmt"
"sync"
"time"
)</p><p>var (
buffer = make([]int, 0, 10)
cond = sync.NewCond(&sync.Mutex{})
isEmpty = true
)</p><p>func producer() {
for i := 0; i < 5; i++ {
cond.L.Lock()
buffer = append(buffer, i)
isEmpty = false
fmt.Printf("生产者: 添加 %d\n", i)
cond.L.Unlock()
cond.Broadcast() // 唤醒所有消费者
time.Sleep(500 * time.Millisecond)
}
}</p><p>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: 消费 %d\n", name, item)
cond.L.Unlock()
time.Sleep(800 * time.Millisecond)
}
}</p><p>func main() {
go producer()
go consumer("消费者A")
go consumer("消费者B")</p><pre class='brush:php;toolbar:false;'>time.Sleep(6 * time.Second)}
在这个例子中,消费者在缓冲区为空时调用 Wait() 进入等待,生产者每次添加数据后调用 Broadcast() 通知所有消费者。Broadcast 能确保至少一个消费者被唤醒处理新数据。
如果只有一个协程在等待某个条件,使用 Signal() 更高效。如果有多个等待者,或不确定等待数量,应使用 Broadcast()。例如,在上述消费者模型中,多个消费者都在等待数据,使用 Broadcast 可以唤醒所有等待者,由它们竞争获取数据。
基本上就这些。sync.Cond 不复杂但容易忽略细节,关键是配合锁使用、在循环中检查条件,并合理选择唤醒方式。
以上就是Golang如何使用sync.Cond实现条件同步的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号