技术文摘
Go协程实现等待多个协程完成的方法
2025-01-09 01:29:40 小编
Go协程实现等待多个协程完成的方法
在Go语言的并发编程中,经常会遇到启动多个协程并行执行任务,然后需要等待所有协程都完成后再进行后续操作的场景。掌握有效的方法来等待多个协程完成,对于构建高效、可靠的并发程序至关重要。
一种常用的方式是使用 sync.WaitGroup。sync.WaitGroup 内部维护着一个计数器,通过 Add 方法增加计数,Done 方法减少计数,Wait 方法则会阻塞当前协程,直到计数器归零。例如,我们有一个需求,需要启动多个协程去计算数组中每个元素的平方,代码如下:
package main
import (
"fmt"
"sync"
)
func squareArray(arr []int, wg *sync.WaitGroup) {
defer wg.Done()
for i := range arr {
arr[i] = arr[i] * arr[i]
}
}
func main() {
var wg sync.WaitGroup
numbers := []int{1, 2, 3, 4, 5}
numCoroutines := 2
partSize := len(numbers) / numCoroutines
for i := 0; i < numCoroutines; i++ {
wg.Add(1)
start := i * partSize
end := (i + 1) * partSize
if i == numCoroutines-1 {
end = len(numbers)
}
go squareArray(numbers[start:end], &wg)
}
wg.Wait()
fmt.Println(numbers)
}
在上述代码中,我们启动了多个协程来处理数组的不同部分,每个协程完成任务后通过 wg.Done() 通知 WaitGroup。主协程调用 wg.Wait() 等待所有协程完成。
另一种方法是利用 sync.Cond 结合 sync.Mutex。sync.Cond 可以在满足特定条件时通知等待的协程。不过相比 sync.WaitGroup,这种方式实现起来更为复杂,常用于需要更精细控制的场景。
还有一种基于通道(Channel)的方式。可以创建一个信号通道,每个协程完成任务后向通道发送一个信号,主协程通过接收通道信号来判断所有协程是否完成。
package main
import (
"fmt"
)
func squareArrayChannel(arr []int, signal chan struct{}) {
defer func() {
signal <- struct{}{}
}()
for i := range arr {
arr[i] = arr[i] * arr[i]
}
}
func main() {
numbers := []int{1, 2, 3, 4, 5}
numCoroutines := 2
signal := make(chan struct{}, numCoroutines)
partSize := len(numbers) / numCoroutines
for i := 0; i < numCoroutines; i++ {
start := i * partSize
end := (i + 1) * partSize
if i == numCoroutines-1 {
end = len(numbers)
}
go squareArrayChannel(numbers[start:end], signal)
}
for i := 0; i < numCoroutines; i++ {
<-signal
}
fmt.Println(numbers)
}
不同的方法适用于不同的场景,开发者可根据实际需求灵活选择,以实现高效且稳定的并发程序。
- 五年程序员竟用 forEach 遍历删除 ArrayList 元素?
- Netflix Conductor:13k 微服务编排引擎推荐
- ICASSP 2024:字节跳动流媒体音频团队创新方案攻克丢包补偿与通用音质修复难题
- 六款神级 PyCharm 高效插件 助力编程腾飞
- Rust 编程基础:条件表达式与循环
- YAML:简单易读的数据序列化格式
- IntelliJ IDEA 代码质量提升的高效插件
- Go 语言之父:开源 14 年,Go 不止是编程语言,成功秘诀何在?
- Go 语言中 init 函数的常见误用
- 摆脱前端框架的 PUA !
- Golang 流水线设计模式的实践探索
- Java 编程中记录日志的十大技巧
- Helm 是什么?怎样提升云原生应用私有化部署效率
- 在错误中学习:洞悉 Go 编程的六大坏习惯
- Fiber 中的请求与响应处理