首页 游戏 软件 资讯 排行榜 专题
首页
编程语言
Go并发控制errgroup.Group的实现示例

Go并发控制errgroup.Group的实现示例

热心网友
36
转载
2026-05-05

errgroup:不止于同步,更优雅地处理并发错误

在Go语言的并发世界里,errgroup 是一个绕不开的利器。它来自 golang.org/x/sync/errgroup,核心使命是为一组执行共同子任务的协程(goroutines)提供同步、错误传播和上下文取消的能力。简单说,它让管理一群“工人”并处理他们可能带来的“麻烦”变得井井有条。

免费影视、动漫、音乐、游戏、小说资源长期稳定更新! 👉 点此立即查看 👈

从 WaitGroup 到 errGroup:解决了什么痛点?

但凡需要等待多个并发任务全部完成后再继续的场景,开发者们第一时间想到的往往是 sync.WaitGroup。它确实是个可靠的同步原语,但有个明显的短板:它只管“等”,不管“错”。任何一个goroutine内部发生的错误,都无法通过WaitGroup本身反馈给主调方。

这时,errgroup.Group 的价值就凸显出来了。你可以把它看作是WaitGroup的“增强版”,它在提供同步等待能力的基础上,额外增加了两大法宝:对返回错误任务的处理能力,以及限制协程并发数的能力。其使用方法与WaitGroup一脉相承,但内部封装了Add和Wait,巧妙地解决了错误无法返回的难题。

来看一个基础示例,感受一下它是如何工作的:

package main
import (
 "fmt"
 "time"
 "golang.org/x/sync/errgroup"
)
func main() {
 g := &errgroup.Group{}
 for i := 0; i < 5; i++ {
  index := i
  g.Go(func() error {
   fmt.Printf("start to execute the %d gorouting\n", index)
   time.Sleep(time.Duration(index) * time.Second)
   if index%2 == 0 {
    return fmt.Errorf("something has failed on grouting:%d", index)
   }
   fmt.Printf("gorouting:%d end\n", index)
   return nil
  })
 }
 if err := g.Wait(); err != nil {
  fmt.Println(err)
 }
}
// Output:
// start to execute the 4 gorouting
// start to execute the 1 gorouting
// start to execute the 0 gorouting
// start to execute the 2 gorouting
// start to execute the 3 gorouting
// gorouting:1 end
// gorouting:3 end
// something has failed on grouting:0

运行这段代码,你会发现一个关键特性:当多个goroutine出错时,errgroup只会返回第一个出错的goroutine的错误信息。这符合快速失败(fail-fast)的常见设计原则。另一个需要留意的点是,无论是否有协程执行失败,Wait()方法都会忠实地等待所有协程执行完毕,确保资源被妥善清理。

进阶特性:上下文与并发控制

除了基础功能,errgroup还提供了更贴合现代Go开发范式的特性。首先是原生支持context,方便进行跨协程的取消信号传播和超时控制:

g, _ := errgroup.WithContext(context.Background()) // 支持 context

其次,Wait()方法的设计考虑到了复用性,它可以被多次调用,并且每次都能得到相同的错误信息,这为一些需要重复检查结果的场景提供了便利:

 ...
 if err := g.Wait(); err != nil {
  fmt.Println(err)
 }
 if err := g.Wait(); err != nil { // 可再次调用 Wait,依然可以得到 group 的 error 信息
  fmt.Println(err)
 }

核心利器:限制最大并发数

在实际生产环境中,无限制地创建goroutine可能导致资源耗尽。errgroup的SetLimit()方法正是为此而生。它用于限制该组中同时处于活动状态(即正在处理业务)的goroutine的最大数量

通过一个简单的任务处理示例,可以清晰地看到它的效果:

package main
import (
 "log"
 "time"
 "golang.org/x/sync/errgroup"
)
func main() {
 jobs := make(chan int, 10)
 go func() {
  for i := 0; i < 8; i++ {
   jobs <- i + 1
  }
  close(jobs)
 }()
 eg:= &errgroup.Group{}
 eg.SetLimit(3)
 for j := range jobs {
  j := j
  eg.Go(func() error {
   log.Printf("handle job: %d\n", j)
   time.Sleep(2 * time.Second)
   return nil
  })
 }
 eg.Wait()
}

在这个例子中,我们创建了8个任务,但通过SetLimit(3),确保了同一时间最多只有3个goroutine在忙碌地处理任务。

Output:

2024/12/17 18:28:19 handle job: 3

2024/12/17 18:28:19 handle job: 1

2024/12/17 18:28:19 handle job: 2

2024/12/17 18:28:21 handle job: 4

2024/12/17 18:28:21 handle job: 6

2024/12/17 18:28:21 handle job: 5

2024/12/17 18:28:23 handle job: 7

2024/12/17 18:28:23 handle job: 8

仔细观察输出结果的时间戳,规律一目了然:任务1、2、3同时开始,2秒后任务4、5、6接棒,再2秒后任务7、8最后完成。这完美印证了并发数被限制在3个。其内部实现原理并不复杂,本质上是使用了一个带缓冲的channel作为计数信号量(counting semaphore)来控制并发许可的发放,这种实现既高效又简洁。

话说回来,将错误处理、并发控制和同步等待三者优雅地结合,正是errgroup在Go并发工具箱中占据一席之地的原因。对于需要精细化管理并发任务流的场景,它无疑是一个值得深入掌握的标准组件。

来源:https://www.jb51.net/jiaoben/362176tx6.htm
免责声明: 游乐网为非赢利性网站,所展示的游戏/软件/文章内容均来自于互联网或第三方用户上传分享,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系youleyoucom@outlook.com。

最新APP

宝宝过生日
宝宝过生日
应用辅助 04-07
台球世界
台球世界
体育竞技 04-07
解绳子
解绳子
休闲益智 04-07
骑兵冲突
骑兵冲突
棋牌策略 04-07
三国真龙传
三国真龙传
角色扮演 04-07

热门推荐

红米Note11 Pro更新系统需连WiFi吗?
电脑教程
红米Note11 Pro更新系统需连WiFi吗?

红米Note 11 Pro系统升级,为何坚持要求连接Wi-Fi? 当红米Note 11 Pro收到MIUI或澎湃OS的系统更新推送时,官方总会明确提示:整个过程请在Wi-Fi网络环境下完成。这项要求并非随意设定,而是基于清晰的技术与体验考量。一次完整的系统升级包,其大小通常在2GB至4GB之间。如果

热心网友
05.05
小米13ultra有nfc功能吗
电脑教程
小米13ultra有nfc功能吗

小米13 Ultra的NFC功能深度解析:它如何重新定义“全场景智能交互”? 在旗舰手机领域,NFC功能看似已成为标配,但体验却千差万别。小米13 Ultra所搭载的全功能NFC方案,在“全能”与“好用”两个维度上树立了新的标杆。它不仅无缝集成了公交卡模拟、门禁卡复制、数字车钥匙等核心生活服务,更全

热心网友
05.05
嵌入式消毒柜电源插座位置必须外露吗?
电脑教程
嵌入式消毒柜电源插座位置必须外露吗?

嵌入式消毒柜电源插座安装指南:隐蔽式布局提升安全与美观 在规划嵌入式消毒柜的安装方案时,电源插座的布局方式直接影响到最终的整体效果与安全性。正确的做法是避免插座外露,采用隐蔽式安装。根据国家《住宅厨房设计规范》及主流厨电品牌的安装标准,推荐将插座预留在消毒柜后方或侧方的墙体内部,安装高度宜控制在距地

热心网友
05.05
魔音耳机操作说明包含充电指示吗?
电脑教程
魔音耳机操作说明包含充电指示吗?

是的,魔音(Beats)耳机充电状态一目了然,指示灯明确显示 当你为Beats头戴式耳机充电时,如何判断它是否已经充满?答案就藏在机身自带的五段式LED电量指示灯里。在充电过程中,这排指示灯会持续闪烁,实时反馈充电进度。一旦所有五个指示灯全部转为稳定常亮、不再闪烁,即代表电池已完全充满。整个充电周期

热心网友
05.05
博朗剃须刀如何识别型号?
电脑教程
博朗剃须刀如何识别型号?

博朗剃须刀型号全解析:从编码规则到选购技巧的终极指南 面对博朗剃须刀复杂的字母数字组合感到困惑?实际上,其型号命名体系逻辑严谨,是用户选购的核心依据。简单来说,型号首位的数字(1、3、5、7、9)直接代表产品系列,数字越大,通常意味着技术越先进、功能越全面、定位越高端。例如,顶级的9系旗舰机型普遍搭

热心网友
05.05