0

0

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

P粉602998670

P粉602998670

发布时间:2025-11-25 18:42:31

|

574人浏览过

|

来源于php中文网

原创

使用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

func worker() { for i := 0; i < 1000; i++ { mu.Lock() counter++ mu.Unlock() } }

每次修改 counter 前都先加锁,操作完成后释放锁,确保同一时间只有一个goroutine能修改该变量。

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

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

WaitGroup 用于主线程等待一组goroutine执行完毕,常用于批量任务场景。

用法要点:Add增加计数,Done表示完成,Wait阻塞直到计数归零。

var wg sync.WaitGroup

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) }

wg.Wait() fmt.Println("所有工作已完成")

主函数调用 Wait() 后会暂停,直到每个goroutine都调用 Done(),保证任务全部结束再继续。

Pebblely
Pebblely

AI产品图精美背景添加

下载

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

Go提倡“通过通信共享内存,而不是通过共享内存通信”。channel是天然的同步机制

无缓冲channel的发送和接收是同步的,即发送方会阻塞直到有人接收。

done := make(chan bool)

go func() { fmt.Println("执行耗时任务...") time.Sleep(2 * time.Second) done <- true }()

fmt.Println("等待任务完成") <-done fmt.Println("任务完成,继续执行")

这种方式比 sleep 或轮询更高效,还能传递状态信息。

对于多个goroutine的结果收集,可以使用带缓冲的channel配合关闭机制:

resultCh := make(chan int, 3)
for i := 0; i < 3; i++ {
    go func(num int) {
        resultCh <- num * num
    }(i)
}

for i := 0; i < 3; i++ { result := <-resultCh fmt.Println("结果:", result) }

使用 sync.Once 实现单次初始化

某些场景下需要确保某段逻辑只执行一次,例如配置加载、连接初始化等。sync.Once 提供了线程安全的保障。

var once sync.Once
var config map[string]string

func loadConfig() { once.Do(func() { config = make(map[string]string) config["api_url"] = "https://www.php.cn/link/710ba53b0d353329706ee1bedf4b9b39" fmt.Println("配置已加载") }) }

// 多个goroutine调用,只会加载一次 go loadConfig() go loadConfig()

无论多少次调用 loadConfig,内部初始化代码仅执行一次。

基本上就这些。合理使用 Mutex、WaitGroup、channel 和 Once,就能应对大多数goroutine同步需求。关键是理解每种工具的适用场景:保护数据用 Mutex,等任务完成用 WaitGroup,传消息用 channel,做初始化用 Once。不复杂但容易忽略细节,写并发程序时多用 -race 检测竞态问题。

相关专题

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

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

180

2024.02.23

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

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

228

2024.02.23

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

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

340

2024.02.23

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

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

209

2024.03.05

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

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

393

2024.05.21

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

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

197

2025.06.09

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

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

191

2025.06.10

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

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

233

2025.06.17

AO3中文版入口地址大全
AO3中文版入口地址大全

本专题整合了AO3中文版入口地址大全,阅读专题下面的的文章了解更多详细内容。

1

2026.01.21

热门下载

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

精品课程

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

共18课时 | 4.7万人学习

TypeScript 教程
TypeScript 教程

共19课时 | 2.3万人学习

C++教程
C++教程

共115课时 | 13.1万人学习

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

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