首页 > 后端开发 > Golang > 正文

Golang如何处理goroutine同步问题_Golang goroutine同步实践

P粉602998670
发布: 2025-11-25 18:42:31
原创
556人浏览过
使用sync.Mutex、WaitGroup和channel可实现goroutine同步:Mutex保护共享资源避免竞态,WaitGroup等待所有任务完成,channel通过通信实现安全数据传递与协程控制。

golang如何处理goroutine同步问题_golang goroutine同步实践

在Go语言中,goroutine是实现并发的核心机制。但多个goroutine同时运行时,如何保证它们之间的数据一致性和执行顺序,就成了必须面对的问题。处理好goroutine的同步,才能避免竞态条件(race condition)和数据混乱。

使用 sync.Mutex 保护共享资源

当多个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语言免费学习笔记(深入)”;

使用 sync.WaitGroup 等待所有goroutine完成

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(),保证任务全部结束再继续。

Clipfly
Clipfly

一站式AI视频生成和编辑平台,提供多种AI视频处理、AI图像处理工具。

Clipfly 129
查看详情 Clipfly

使用 channel 进行goroutine间通信与同步

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 实现单次初始化

某些场景下需要确保某段逻辑只执行一次,例如配置加载、连接初始化等。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中文网其它相关文章!

最佳 Windows 性能的顶级免费优化软件
最佳 Windows 性能的顶级免费优化软件

每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。

下载
来源:php中文网
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新 English
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习

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