使用sync.Mutex、WaitGroup和channel可实现goroutine同步:Mutex保护共享资源避免竞态,WaitGroup等待所有任务完成,channel通过通信实现安全数据传递与协程控制。

在Go语言中,goroutine是实现并发的核心机制。但多个goroutine同时运行时,如何保证它们之间的数据一致性和执行顺序,就成了必须面对的问题。处理好goroutine的同步,才能避免竞态条件(race condition)和数据混乱。
当多个goroutine访问同一变量或结构体时,必须通过锁机制来防止并发修改。sync.Mutex 是最常用的互斥锁工具。
比如多个goroutine同时对一个计数器进行加操作:
var counter int
var mu sync.Mutex
<p>func worker() {
for i := 0; i < 1000; i++ {
mu.Lock()
counter++
mu.Unlock()
}
}</p>每次修改 counter 前都先加锁,操作完成后释放锁,确保同一时间只有一个goroutine能修改该变量。
立即学习“go语言免费学习笔记(深入)”;
WaitGroup 用于主线程等待一组goroutine执行完毕,常用于批量任务场景。
用法要点:Add增加计数,Done表示完成,Wait阻塞直到计数归零。
var wg sync.WaitGroup
<p>for i := 0; i < 5; i++ {
wg.Add(1)
go func(id int) {
defer wg.Done()
fmt.Printf("worker %d starting\n", id)
time.Sleep(time.Second)
fmt.Printf("worker %d done\n", id)
}(i)
}</p><p>wg.Wait()
fmt.Println("所有工作已完成")</p>主函数调用 Wait() 后会暂停,直到每个goroutine都调用 Done(),保证任务全部结束再继续。
Go提倡“通过通信共享内存,而不是通过共享内存通信”。channel是天然的同步机制。
无缓冲channel的发送和接收是同步的,即发送方会阻塞直到有人接收。
done := make(chan bool)
<p>go func() {
fmt.Println("执行耗时任务...")
time.Sleep(2 * time.Second)
done <- true
}()</p><p>fmt.Println("等待任务完成")
<-done
fmt.Println("任务完成,继续执行")</p>这种方式比 sleep 或轮询更高效,还能传递状态信息。
对于多个goroutine的结果收集,可以使用带缓冲的channel配合关闭机制:
resultCh := make(chan int, 3)
for i := 0; i < 3; i++ {
go func(num int) {
resultCh <- num * num
}(i)
}
<p>for i := 0; i < 3; i++ {
result := <-resultCh
fmt.Println("结果:", result)
}</p>某些场景下需要确保某段逻辑只执行一次,例如配置加载、连接初始化等。sync.Once 提供了线程安全的保障。
var once sync.Once
var config map[string]string
<p>func loadConfig() {
once.Do(func() {
config = make(map[string]string)
config["api_url"] = "<a href="https://www.php.cn/link/710ba53b0d353329706ee1bedf4b9b39">https://www.php.cn/link/710ba53b0d353329706ee1bedf4b9b39</a>"
fmt.Println("配置已加载")
})
}</p><p>// 多个goroutine调用,只会加载一次
go loadConfig()
go loadConfig()</p>无论多少次调用 loadConfig,内部初始化代码仅执行一次。
基本上就这些。合理使用 Mutex、WaitGroup、channel 和 Once,就能应对大多数goroutine同步需求。关键是理解每种工具的适用场景:保护数据用 Mutex,等任务完成用 WaitGroup,传消息用 channel,做初始化用 Once。不复杂但容易忽略细节,写并发程序时多用 -race 检测竞态问题。
以上就是Golang如何处理goroutine同步问题_Golang goroutine同步实践的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号