go 中的WaitGroup
wait_group
sync.WaitGroup 类型是并发安全的,也是开箱就能用的。 该类型有三个指针方法,即:Add,Done和Wait.
sync.WaitGroup是一个结构体类型。其中一个代表计数的字节数组类型的字段,该 字段用4字节表示给定计数,另4字节表示等待计数。当一个sync.WaitGroup类型的变 量被声明之后,其中的这两个计数都会是0。可以通过add方法增大或减少其中给定计数, 例如:
wg.Add(3)
或
wg.Add(-3)
需要注意的是,我们不能让这个计数值变成一个负数。
wg.Done()
相当于
wg.Add(-1)
我们知道add和done方法可以变更计数器的值,但是变更之后具体有什么 作用呢?
当调用sync.WaitGroup类型值的Wait方法时,它会去检查给定计数。如果 该计数为0,那么该方法会立即返回,且不会对程序产生任何影响。但是, 如果这个计数器大于0,该方法调用所在的那个goroutine就会阻塞,同时 等待计数器会加1。直到在该值的add或done方法被调用时给定计数变回0. 该值才会去唤醒因此等待而阻塞的所有goroutine,同时清零等待计数。
现在我们有一个案例: 假设程序启用了4个goroutine,分别是g1,g2,g3,g4。其中g2,g3,g4是由代码g1 启用的,g1启用之后并且要等待这些特殊任务的完成。
使用通道来进行阻塞
sign := make(chan int, 3) go func() { sign <- 2 fmt.Println(2) }() go func() { sign <- 3 fmt.Println(3) }() go func() { sign <- 4 fmt.Println(4) }() for i := 0; i < 3; i++ { fmt.Println("执行", <-sign) }
使用通道的有过于繁重了,原则上,我们不应该把通道当做互斥锁或信号量来使用。
使用waitGroup
var wg sync.WaitGroup wg.Add(3) go func() { wg.Done() fmt.Println(2) }() go func() { wg.Done() fmt.Println(3) }() go func() { wg.Done() fmt.Println(4) }() wg.Wait() fmt.Println("1 2 3 4 end")
参考:go并发编程实战
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
· 开发者必知的日志记录最佳实践
· SQL Server 2025 AI相关能力初探
· Linux系列:如何用 C#调用 C方法造成内存泄露
· AI与.NET技术实操系列(二):开始使用ML.NET
· 没有Manus邀请码?试试免邀请码的MGX或者开源的OpenManus吧
· 无需6万激活码!GitHub神秘组织3小时极速复刻Manus,手把手教你使用OpenManus搭建本
· C#/.NET/.NET Core优秀项目和框架2025年2月简报
· DeepSeek在M芯片Mac上本地化部署
· 葡萄城 AI 搜索升级:DeepSeek 加持,客户体验更智能