0

0

如何在Golang中处理goroutine间依赖_Golang channel同步与依赖控制实践

P粉602998670

P粉602998670

发布时间:2025-12-30 19:02:02

|

706人浏览过

|

来源于php中文网

原创

用 sync.WaitGroup 等待 goroutine 完成:Add(1) 必须在 go 前调用,Done() 推荐 defer 调用;需多个 goroutine 但只取首个结果时,用 select 监听同类型 channel 实现“first result wins”。

如何在golang中处理goroutine间依赖_golang channel同步与依赖控制实践

goroutine 启动后怎么等它做完再继续?

sync.WaitGroup 最直接。它不传数据,只管“有几个 goroutine 在跑、它们是否都结束了”。WaitGroupAdd 必须在 goroutine 启动前调用,否则可能 Wait 永远阻塞或 panic。

  • Add(1) 要在 go func() {...}() 之前,不能写在 goroutine 内部
  • 每个 goroutine 结束前必须调用 Done(),推荐用 defer wg.Done() 防遗漏
  • 不要重复 Add 同一个 WaitGroup 多次却不 Done,会卡死
var wg sync.WaitGroup
wg.Add(1)
go func() {
    defer wg.Done()
    time.Sleep(100 * time.Millisecond)
    fmt.Println("done")
}()
wg.Wait() // 主协程在这里阻塞,直到上面的 goroutine 调用 Done

需要等多个 goroutine,但只关心“第一个返回的结果”?

select + chan 配合 defaultcontext.WithCancel 控制超时,但更常见的是用 select 监听多个 chan,谁先发就收谁——这就是 “first result wins” 模式。

  • 所有结果 channel 必须是同类型,比如 chan int
  • 避免漏收:没被选中的 channel 会一直挂着,如果后续还要读,得确保有 goroutine 在持续写或关闭它
  • 如果只想要第一个,其余 goroutine 应该能及时退出,否则浪费资源;可用 context.Context 通知取消
ch1 := make(chan string, 1)
ch2 := make(chan string, 1)
go func() { ch1 <- "from A" }()
go func() { ch2 <- "from B" }()

select { case s := <-ch1: fmt.Println(s) case s := <-ch2: fmt.Println(s) }

goroutine A 必须等 goroutine B 发出信号才能开始?

用无缓冲 channel(make(chan struct{}))做“信号量”最轻量。它不传业务数据,只起同步作用,零内存开销,语义清晰。

  • 发送方用 ch ,接收方用 阻塞等待
  • 无缓冲 channel 的 send 和 receive 必须配对,否则会死锁;建议用 defer 确保发送一定发生
  • 别用 intbool 类型 channel 传信号——容易误读意图,也浪费空间
ready := make(chan struct{})
go func() {
    time.Sleep(50 * time.Millisecond)
    close(ready) // 或用 ready <- struct{}{}
}()
<-ready // 主协程在此阻塞,直到另一端发出信号
fmt.Println("now start")

channel 关闭后还能读吗?读完会怎样?

能读,但行为分两种:带 ok 判断的接收会返回零值 + false;不带判断的接收也会返回零值,但无法区分是“刚关闭”还是“还没写入”。这是最容易踩的坑。

Bika.ai
Bika.ai

打造您的AI智能体员工团队

下载

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

  • 关闭 channel 前确保没人再往里写,否则 panic:send on closed channel
  • 多个 goroutine 从同一 channel 读没问题,但关闭只能由写方决定,且只能关一次
  • for range ch 自动处理关闭逻辑,比手动 更安全
ch := make(chan int, 2)
ch <- 1
ch <- 2
close(ch)

v, ok := <-ch // v==1, ok==true v, ok = <-ch // v==2, ok==true v, ok = <-ch // v==0, ok==false ← 注意:v 是 int 零值,不是错误

实际依赖控制中,WaitGroupchan struct{} 解决“是否完成”,select 解决“谁先完成”,而 channel 关闭状态的误判,往往出现在没有统一读取约定的多人协作代码里。

相关专题

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

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

173

2024.02.23

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

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

224

2024.02.23

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

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

335

2024.02.23

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

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

206

2024.03.05

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

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

388

2024.05.21

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

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

193

2025.06.09

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

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

187

2025.06.10

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

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

191

2025.06.17

桌面文件位置介绍
桌面文件位置介绍

本专题整合了桌面文件相关教程,阅读专题下面的文章了解更多内容。

0

2025.12.30

热门下载

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

精品课程

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

共32课时 | 3.1万人学习

Go语言实战之 GraphQL
Go语言实战之 GraphQL

共10课时 | 0.8万人学习

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

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