标签 并发 下的文章

WaitGroup.Go要来了?Go官方提案或让你告别Add和Done样板代码

本文永久链接 – https://tonybai.com/2025/04/03/waitgroup-go-proposal

sync.WaitGroup是Go语言中处理并发任务同步最常用的原语之一。然而,其经典的Add(1)、go func() { defer wg.Done() … }()、Wait()模式虽然强大,却也因其固定写法和潜在的陷阱(如忘记Done或将Add误置于goroutine内部)而让开发者时常感到繁琐,对新手尤其不友好。近日,一项旨在简化这一模式的提案#63796在Go社区引发了广泛关注,并已被标记为Likely Accept,预示着sync.WaitGroup可能很快将迎来一个实用的新方法:Go。这也意味着Go开发者可以告别Add、defer Done的样板代码,并避免它们的“陷阱”可能导致的难以捕捉的代码错误。在这篇文章中,我就来简单介绍一下WaitGroup.Go这个提案。

1. 现有模式的痛点与WaitGroup.Go的提出

当前使用WaitGroup的标准模式通常如下所示:

package main

import (
    "fmt"
    "sync"
    "time"
)

func work(id int) {
    fmt.Printf("Worker %d starting\n", id)
    time.Sleep(time.Second)
    fmt.Printf("Worker %d done\n", id)
}

func main() {
    var wg sync.WaitGroup
    for i := 1; i <= 5; i++ {
        // 注意:在 Go 1.22 之前,需要 i := i 来避免闭包捕获问题
        // i := i
        wg.Add(1) // 必须在启动 goroutine 前调用 Add
        go func(id int) {
            defer wg.Done() // 必须在 goroutine 退出前调用 Done
            work(id)
        }(i)
    }
    wg.Wait() // 等待所有 goroutine 完成
    fmt.Println("All workers done")
}

这种样板使用模式存在几个容器出错的关键点:

  • wg.Add(1) 的位置: 必须在启动goroutine之前调用。如果将其放在goroutine内部,可能会导致Wait在Add执行前就返回,引发panic或竞态条件。这是最常见的错误之一。
  • defer wg.Done(): 必须确保在goroutine逻辑结束时调用Done,否则Wait将永久阻塞。defer是推荐做法,但也可能被遗漏。
  • 闭包变量捕获 (Go < 1.22):Go 1.22之前的版本中,循环变量直接在goroutine的闭包中使用会导致所有goroutine共享同一个变量值,需要i := i 这样的技巧来创建副本。

为了解决这些问题,提案#63796 建议为sync.WaitGroup添加一个Go方法:

// Go calls f on a new goroutine and adds that task to the WaitGroup.
// When f returns, the task is removed from the WaitGroup.
// ... (其他文档细节省略)
func (wg *WaitGroup) Go(f func()) {
    wg.Add(1)
    go func() {
        defer wg.Done()
        f()
    }()
}

这个方法简洁地封装了Add(1)、启动goroutine和defer Done()的逻辑。使用Go方法后,之前的例子可以大幅简化为下面代码:

package main

import (
    "fmt"
    "sync"
    "time"
)

func work(id int) {
    fmt.Printf("Worker %d starting\n", id)
    time.Sleep(time.Second)
    fmt.Printf("Worker %d done\n", id)
}

func main() {
    var wg sync.WaitGroup // 假设WaitGroup已包含Go方法
    for i := 1; i <= 5; i++ {
        // Go 1.22+版本无需i := i
        wg.Go(func() {
            work(i)
        })
    }
    wg.Wait()
    fmt.Println("All workers done")
}

我们可以看到,代码不仅行数减少,而且显著降低了出错的可能性,尤其是避免了Add位置错误这一高频陷阱。

2. 时机成熟:为何现在是引入WaitGroup.Go的好时机?

该提案并非首次提出(相关讨论可追溯至#18022#39863),但之前的提案因各种原因未能被接受。此次能够获得”Likely Accept”的状态,可能主要得益于以下几个因素:

  • Go 1.22循环变量语义变化

Go 1.22正式“修正”了for循环的变量语义,使得每次迭代都会创建新的循环变量实例。这极大地降低了在wg.Go的闭包函数中直接使用循环变量的风险,使得func()形式的API更加安全和自然。正如dsnet在评论中指出的,虽然闭包仍可能引入其他变量修改的风险,但相比wg.Add位置错误,这种风险出现的频率要低得多。

  • 社区实践的验证

许多流行的第三方库(如tailscale.com/syncs和sourcegraph/conc)以及golang.org/x/sync/errgroup都已经实现了类似的Go方法,证明了其在实际开发中的价值和受欢迎程度。这为标准库采纳该模式提供了有力佐证。

  • 错误预防的迫切性

尽管社区曾讨论过通过vet工具检查wg.Add误用(#18022),但此前相关检查迟迟未能落地(直到最近才由adonovan等人推动并合并了相关分析器)。直接在API层面提供更安全的替代方案,被认为是更有效的解决途径。GitHub代码搜索也显示,虽然正确用法占绝大多数,但错误用法(go之后才Add)数量仍然不可忽视(上千例)。

3. 社区讨论焦点

在提案的讨论过程中,社区成员也提出了一些值得思考的问题,这里也找出一些典型的问题供大家玩味:

  • 是否需要新类型?

有人建议创建一个新的类型(如sync.Tasks),以避免WaitGroup同时存在Add/Done和Go两种模式可能带来的混淆。但主流观点认为,将Go方法添加到现有WaitGroup可以方便现有代码的原地升级(gopls甚至已为此添加了自动化重构支持),并且混合使用的风险较低(错误使用Done会快速panic,多余的Add也会导致Wait阻塞,易于发现)。

  • 与errgroup的关系

errgroup.Group也有Go方法,但它还处理了错误传播和context取消。WaitGroup.Go则更纯粹地关注任务同步,两者定位不同,可以共存。将errgroup引入标准库是另一个独立的提案(#57534)。

  • 方法命名

曾有提议使用Start或Run,但Go这个命名与errgroup中的Go保持一致,且能清晰表达“启动新goroutine”的含义,最终获得了更多支持。

  • 文档重塑

Go当前的技术负责人aclements建议将WaitGroup的文档从“计数器”视角转向“任务集合”视角,并将Go作为首选方法进行介绍。对此adonovan提醒WaitGroup本质仍是计数信号量,文档更新需谨慎平衡。

4. 小结

sync.WaitGroup.Go提案的”Likely Accept”状态对于Go开发者来说是一个积极的信号。这个看似简单的补充,有望显著提升Go并发编程的体验,减少Add和Done的样板代码,规避常见错误。它体现了Go团队在保持核心库简洁性的同时,也愿意吸收社区成熟实践、优化开发者体验的务实态度。我们期待在未来的Go版本中看到这一实用特性的正式发布,届时,编写健壮、简洁的并发代码将变得更加容易。

5. 参考资料


Gopher部落知识星球在2025年将继续致力于打造一个高品质的Go语言学习和交流平台。我们将继续提供优质的Go技术文章首发和阅读体验。并且,2025年将在星球首发“Gopher的AI原生应用开发第一课”、“Go陷阱与缺陷”和“Go原理课”专栏!此外,我们还会加强星友之间的交流和互动。欢迎大家踊跃提问,分享心得,讨论技术。我会在第一时间进行解答和交流。我衷心希望Gopher部落可以成为大家学习、进步、交流的港湾。让我相聚在Gopher部落,享受coding的快乐! 欢迎大家踊跃加入!

img{512x368}
img{512x368}

img{512x368}
img{512x368}

著名云主机服务厂商DigitalOcean发布最新的主机计划,入门级Droplet配置升级为:1 core CPU、1G内存、25G高速SSD,价格6$/月。有使用DigitalOcean需求的朋友,可以打开这个链接地址:https://m.do.co/c/bff6eed92687 开启你的DO主机之路。

Gopher Daily(Gopher每日新闻) – https://gopherdaily.tonybai.com

我的联系方式:

  • 微博(暂不可用):https://weibo.com/bigwhite20xx
  • 微博2:https://weibo.com/u/6484441286
  • 博客:tonybai.com
  • github: https://github.com/bigwhite
  • Gopher Daily归档 – https://github.com/bigwhite/gopherdaily
  • Gopher Daily Feed订阅 – https://gopherdaily.tonybai.com/feed

商务合作方式:撰稿、出书、培训、在线课程、合伙创业、咨询、广告合作。

写Go就像喝白开水

本文永久链接 – https://tonybai.com/2024/10/29/go-coding-is-like-drinking-boiled-water

在编程语言的世界里,Go语言简单而直接,它没有复杂的语法和华丽的特性,给人一种纯粹的感觉,让我们在编写代码时感受到了一种清晰和高效。

正如Russ Cox所言,Go的“无聊”恰恰是它的优势。抛开冗余装饰,Go专注于可靠、实用的功能。在这个快节奏的时代,它让我们免受复杂性的困扰,帮助我们快速解决实际问题。

写Go就像喝一杯无味的白开水,虽寡淡却能即刻解渴,满足需求,并少有后顾之忧。平淡中透着从容,是我们日常开发中的可靠之选。

早上打开极客时间的首页,发现我的Go语言第一课专栏在极客时间的7日飙升榜上跃升至第5名(截至2024.10.29 21点),估计这是借了双十一的光,也感谢大家的支持与厚爱!借此在这里给自己的专栏打个广告。

Go语言:简单直接

正如前面所提到的,Go语言的设计理念就是追求简单与直接。无论是基础语法还是并发编程,Go都让你在最短的时间内上手,效果立竿见影。你会发现,编写清晰、优雅的代码并不是一件难事。

Go社区的快速增长

今年,Go语言的社区发展迅猛,这在Reddit的Go分论坛上体现明显,每周都有超过1k名新会员加入。这不仅显示了Go语言的受欢迎程度,更证明了它在开发者中的广泛应用。

号召大家入门Go

如果你还在犹豫,不妨趁这个时机,加入Go的学习行列!无论你是编程新手还是经验丰富的开发者,Go都能为你打开新的大门。我的专栏将为你提供实用的学习资源和案例分析,助你快速入门。

一起踏上Go之旅!

现在就扫码关注我的“Go语言第一课”专栏吧,让我们一起体验Go语言的魅力,享受编程的乐趣!期待在这个快速发展的社区中见到你的身影!

img{512x368}

读者精彩评论

以下是“写Go就像喝白开水”一文在公众号首发后一些读者的精彩评论的摘录:

C是自来水,得烧开才能喝的安全,go是已经烧好的白开水,即时解渴,高效,c#,Java 是奶茶咖啡,看起来高级,但加了狠活,弄不好很难喝,喝完可能窜稀。 — 网友 Run

C++是白酒喝高了上头 — 网友 ニコニコ

boring but useful 形容go确实贴切 — 网友 领个废宅


Gopher部落知识星球在2024年将继续致力于打造一个高品质的Go语言学习和交流平台。我们将继续提供优质的Go技术文章首发和阅读体验。同时,我们也会加强代码质量和最佳实践的分享,包括如何编写简洁、可读、可测试的Go代码。此外,我们还会加强星友之间的交流和互动。欢迎大家踊跃提问,分享心得,讨论技术。我会在第一时间进行解答和交流。我衷心希望Gopher部落可以成为大家学习、进步、交流的港湾。让我相聚在Gopher部落,享受coding的快乐! 欢迎大家踊跃加入!

img{512x368}
img{512x368}

img{512x368}

著名云主机服务厂商DigitalOcean发布最新的主机计划,入门级Droplet配置升级为:1 core CPU、1G内存、25G高速SSD,价格5$/月。有使用DigitalOcean需求的朋友,可以打开这个链接地址:https://m.do.co/c/bff6eed92687 开启你的DO主机之路。

Gopher Daily(Gopher每日新闻) – https://gopherdaily.tonybai.com

我的联系方式:

  • 微博(暂不可用):https://weibo.com/bigwhite20xx
  • 微博2:https://weibo.com/u/6484441286
  • 博客:tonybai.com
  • github: https://github.com/bigwhite
  • Gopher Daily归档 – https://github.com/bigwhite/gopherdaily
  • Gopher Daily Feed订阅 – https://gopherdaily.tonybai.com/feed

商务合作方式:撰稿、出书、培训、在线课程、合伙创业、咨询、广告合作。

如发现本站页面被黑,比如:挂载广告、挖矿等恶意代码,请朋友们及时联系我。十分感谢! Go语言第一课 Go语言进阶课 Go语言精进之路1 Go语言精进之路2 Go语言第一课 Go语言编程指南
商务合作请联系bigwhite.cn AT aliyun.com

欢迎使用邮件订阅我的博客

输入邮箱订阅本站,只要有新文章发布,就会第一时间发送邮件通知你哦!

这里是 Tony Bai的个人Blog,欢迎访问、订阅和留言! 订阅Feed请点击上面图片

如果您觉得这里的文章对您有帮助,请扫描上方二维码进行捐赠 ,加油后的Tony Bai将会为您呈现更多精彩的文章,谢谢!

如果您希望通过微信捐赠,请用微信客户端扫描下方赞赏码:

如果您希望通过比特币或以太币捐赠,可以扫描下方二维码:

比特币:

以太币:

如果您喜欢通过微信浏览本站内容,可以扫描下方二维码,订阅本站官方微信订阅号“iamtonybai”;点击二维码,可直达本人官方微博主页^_^:
本站Powered by Digital Ocean VPS。
选择Digital Ocean VPS主机,即可获得10美元现金充值,可 免费使用两个月哟! 著名主机提供商Linode 10$优惠码:linode10,在 这里注册即可免费获 得。阿里云推荐码: 1WFZ0V立享9折!


View Tony Bai's profile on LinkedIn
DigitalOcean Referral Badge

文章

评论

  • 正在加载...

分类

标签

归档



View My Stats