别再无脑 go func() 了!Go 资深布道师 Dave Cheney 的 Goroutine 管理哲学

本文永久链接 – https://tonybai.com/2026/04/13/dave-cheney-goroutine-management-philosophy

大家好,我是Tony Bai。

在 Go 语言的江湖里,go func() 就像一把绝世好剑。它轻灵、锋利,只需几个字符,就能让你瞬间拥有“分身术”,并发地处理海量任务。Go 团队曾自豪地告诉我们:Goroutine 很廉价,你可以随手启动成千上万个。

于是,我们习惯了在代码里肆意挥洒:

  • HTTP 请求来了?go handle()。
  • 要写日志?go log()。
  • 要发通知?go notify()。
  • … …

我们以为自己掌握了并发的捷径。

但就在去年的 GopherCon Singapore 技术大会上,Go 社区的资深布道师 Dave Cheney,却用一场充满哲学思考的演说,给所有 Gopher 敲响了警钟。

他的核心论点很明确:Goroutine 绝非免费的午餐,它是一种需要付出代价的“有限资源”。如果你只管启动(Start)而不懂如何停止(Stop),你并没有在写高效的并发程序,你只是在为系统埋下慢性自杀的伏笔。

今天,我们就来深度拆解 Dave Cheney 的这场重要演讲,梳理出他在 AI 大模型和微服务时代,为我们总结的 “Goroutine 声明周期管理四大哲学”以及他最终给出的Goroutine管理方案。

哲学一:内存是有价的,而 Goroutine 是“内存之根”

Dave Cheney 在演讲开头提出了一个极其硬核的观点:内存不是无限的,它是和数据库连接、文件句柄一样的有限资源。

在 Java 或 C++ 中,我们要时刻担心内存泄漏。但在 Go 里,我们觉得有 GC(垃圾回收器)在,一切无忧。

然而,Dave 指出了一个被 99% 的人忽略的真相:在 Go 的世界里,每一个正在运行的 Goroutine,都是一个“GC 根节点(GC Root)”。

什么意思?

只要一个 Goroutine 还在运行,它所引用的所有内存、它栈上的所有变量、它指向的所有堆对象,GC 都绝对不敢回收。

“你可以关闭一个文件,可以解锁一个互斥锁。但你如何‘回收’一个失控的 Goroutine?”

如果你启动了一个 Goroutine 后失去了对它的追踪,它就变成了一个永远无法回收的“内存僵尸”。它不仅自己霸占着 2KB 以上的栈空间,更可能死死拽着几个 GB 的业务对象不撒手。

哲学二:永远不要启动一个你不知道如何停止的 Goroutine

这是 Dave Cheney 演讲中最核心的一句军规:Never start a goroutine without knowing how it will stop.

为了证明“野 Goroutine”的破坏力,Dave 在现场演示了一个极其经典的血泪 Demo。

他写了一个 HTTP 服务器,为了让请求秒回,他把日志记录放到了后台:go logRequest(r)。

接着,他通过重定向标准输出模拟了下游日志系统网络拥堵、写入被阻塞的场景。

恐怖的一幕发生了:

服务器内存开始疯狂飙升,每秒钟都有成百上千个新的 Goroutine 被创建,但因为输出被阻塞,它们全都卡在写入的那一行,一个都死不掉。
不到一分钟,整个程序因为 OOM(内存溢出)当场暴毙。

Dave 的结论非常冷酷:

启动一个 Goroutine 只需要 1 微秒,但如果不考虑它的“死法”,这个 Goroutine 最终会成为杀掉你整个集群的凶手。

哲学三:不要强迫它停,要“优雅地求它停”

在 Java 中,曾经有一个 thread.stop() 方法,后来被禁用了,因为它会引发不可控的资源损坏。Go 语言聪明地避开了这个坑:Go 没有任何一种方式,能让一个 Goroutine 强行停止另一个。

你只能通过 “协同(Cooperation)”

Dave 强调,defer 是 Goroutine 的“临终遗言”。所有的资源释放(文件关闭、锁解除)都必须放在 defer 里。

而管理这一切的唯一“生死符”,就是 Context

在 Dave 的哲学里,一个合格的后台服务函数,必须长成这样:

func (s *Service) Run(ctx context.Context) error {
    // 1. 临终遗言:无论如何,最后一定要清理战场
    defer s.cleanup() 

    for {
        select {
        case <-ctx.Done():
            // 2. 收到“生死符”,优雅退出
            return ctx.Err()
        case task := <-s.taskChan:
            s.process(task)
        }
    }
}

你必须给 Goroutine 一个“想得开”的机会,让它在收到 ctx.Done() 时,带着所有的 defer 体面地离开。

哲学四:把并发权留给调用者,而不是库

这是 Dave Cheney 给库开发者(Library Authors)提出的最高阶要求。

他引用了另一位大神 Peter Bourgon 的话:“Leave concurrency to the caller.”

一个设计糟糕的库: 在你调用 NewProvider() 的时候,悄悄在后台启动了一个 Goroutine 去跑心跳,却没给你返回任何停止它的句柄。这种库是不可靠的。

一个具有“管理哲学”的库: 即使它需要后台运行,它也应该把那个 Run 函数暴露给用户,让用户自己决定:

  • 是开一个 Goroutine 去跑它?
  • 还是把它扔进一个 errgroup 里集中管控?
  • 还是干脆同步运行它?

只有这样,作为顶层架构师的你,才能真正实现所有子系统的 “同生共死”

历史的挣扎:从 Tomb 到 Errgroup,我们与“失控”的斗争

事实上,Go 社区与“Goroutine 管理”这个恶魔的斗争,从 2012 年就开始了。Dave带着我们一起回顾了一下社区的方案,虽然每个方案都不完美!

第一代武器:Tomb (坟墓)

来自 Canonical(Ubuntu 母公司)的 Juju 项目,发明了 tomb 包。它通过一个 t.Go() 方法来启动 Goroutine,并用一个 t.Wait() 来等待它们全部结束。但它的缺点是,如何通知这些 Goroutine“你们该停了”,依然需要开发者手动传来传去。

第二代武器:Errgroup

由 Go 社区大神 Brad Fitzpatrick 编写的 errgroup,极大地简化了“并发执行一组任务,并收集第一个错误”的场景。但它同样没有解决“如何优雅地通知所有任务提前中止”的问题。

第三代武器:OK Log 的 group 包

由 Peter Bourgon 设计的 group 包,首次引入了一个极其优雅的范式。它要求你在添加一个任务时,必须同时提供两个函数:一个 execute 函数(如何启动),和一个 interrupt 函数(如何打断)。

这是一种“契约式”的设计,强制开发者在启动一个 Goroutine 的时候,就必须想好如何杀死它。

Dave Cheney 的Goroutine管理方案

在吸收了上述哲学以及社区尝试后,Dave 给出了一个现代 Go 微服务的“标准起手式”,当然也是他自己的Goroutine管理方案:pkg/group。

在吸收了社区十几年来的所有经验和教训之后,Dave Cheney 在演讲的最后,亮出了他自己多年来在无数个项目中沉淀下来的“终极武器”——一个同样名为 group 的、集大成的 Goroutine 管理库:pkg/group,也可以认为是一个现代 Go 微服务的“标准起手式”:

在 Dave Cheney 的 group 里,你添加的每一个任务,都必须是一个接受 context.Context 作为参数的函数。

g.Add(func(ctx context.Context) error {
    // ...
})

Context 成了所有 Goroutine 唯一的“生死符”。无论是超时、是上游请求被取消、还是整个服务收到了 SIGTERM 信号准备关闭,都会通过 ctx.Done() 这个唯一的通道,通知到每一个角落。

在 Dave Cheney 的 group 中,任何一个子 Goroutine 发生的 panic,都不会导致整个进程崩溃。它会被 recover 住,转化为一个 error,然后触发整个 group 的优雅关闭流程。

pkg/group的使用典型示例如下:


在这段代码里,所有的后台服务被捆绑成了一个“命运共同体”。任何一个服务失败,或者 k8s 发来关闭 Pod 的信号,都会导致所有服务一起进入优雅关闭流程,确保数据不丢失、连接被妥善断开。

小结

从“启动”到“坟墓”,Dave Cheney 为我们揭示了并发编程的下半场:Goroutine管理

go func() 赋予了我们随手创造并发的权力,但真正体现架构师功力的,是你管理这些并发生命周期的责任感。

下一次,当你在键盘上敲下那几个字符时,请停顿一秒。

想一想:这把剑挥出去,你还能收回来吗?

资料链接:https://www.youtube.com/watch?v=eJLVT157BSs


今日互动探讨:

在你的项目中,是否曾遇到过 Goroutine 泄漏导致的内存灾难?你是如何定位出那个“失踪”的 Goroutine 的?

欢迎在评论区分享你的避坑经验!


还在为“复制粘贴喂AI”而烦恼?我的新专栏 AI原生开发工作流实战 将带你:

  • 告别低效,重塑开发范式
  • 驾驭AI Agent(Claude Code),实现工作流自动化
  • 从“AI使用者”进化为规范驱动开发的“工作流指挥家”

扫描下方二维码,开启你的AI原生开发之旅。


你的Go技能,是否也卡在了“熟练”到“精通”的瓶颈期?

  • 想写出更地道、更健壮的Go代码,却总在细节上踩坑?
  • 渴望提升软件设计能力,驾驭复杂Go项目却缺乏章法?
  • 想打造生产级的Go服务,却在工程化实践中屡屡受挫?

继《Go语言第一课》后,我的《Go语言进阶课》终于在极客时间与大家见面了!

我的全新极客时间专栏 《Tony Bai·Go语言进阶课》就是为这样的你量身打造!30+讲硬核内容,带你夯实语法认知,提升设计思维,锻造工程实践能力,更有实战项目串讲。

目标只有一个:助你完成从“Go熟练工”到“Go专家”的蜕变! 现在就加入,让你的Go技能再上一个新台阶!


商务合作方式:撰稿、出书、培训、在线课程、合伙创业、咨询、广告合作。如有需求,请扫描下方公众号二维码,与我私信联系。

AI 时代,敏捷宣言已死?听听 Martin Fowler 和 Kent Beck 怎么说

本文永久链接 – https://tonybai.com/2026/04/12/agile-manifesto-dead-in-ai-era-martin-fowler-kent-beck

大家好,我是Tony Bai。

25 年前,在美国犹他州的一间滑雪小屋里,17 位当时最顶尖的软件开发者聚集一堂,共同签署了一份将彻底改变未来二十年软件工程形态的纲领——《敏捷软件开发宣言》。

在这 17 位“上古大神”中,有两个名字,如同北极星一般,指引了一代又一代程序员的成长:一位是《重构》的作者 Martin Fowler,另一位则是“极限编程(XP)”之父、敏捷宣言的发起人 Kent Beck。

25 年后的今天,当生成式 AI 的海啸席卷全球,当“敏捷迭代”被 AI 的“瞬间生成”无情碾压时,我们不禁要问:敏捷已死吗?我们曾经信奉的那些工程哲学,还剩下什么?

就在前几天,在一个汇聚了硅谷最火热 AI 创业者的闭门活动上,这两位白发苍苍的“活化石”出人意料地并肩坐到了一起,进行了一场关于 AI 时代的世纪对话

他们没有去鼓吹 AI 带来了多高的效率,反而用一种极其深刻、甚至有些悲观的视角,对当下这场“AI 狂欢”提出了终极拷问。这场对话,值得我们每一个身处其中的技术人,暂停手中飞速生成的代码,静下心来,一字一句地读完。

历史的轮回:AI,不过是又一个“微处理器”

面对台下年轻开发者对 AI 的狂热与恐慌,Kent Beck 的开场异常平静。他把时间拉回到了自己还是个孩子的时候。

“在微处理器(Microprocessor)诞生之前,电脑是一个你根本搬不动的庞然大物。当英特尔 4004 芯片问世时,我们突然意识到,‘等等,这也是一台电脑!’ 突然之间,你能做的事情的想象空间被无限放大了。”

Kent Beck 认为,今天的 AI,在本质上与当年的微处理器、后来的面向对象、再后来的互联网浪潮并无不同。它们都是“想象力的放大器”。

他坦言自己现在正在用 AI 去做一些“极其离谱的、野心勃勃的项目”,比如用 Rust 写库级别的高质量代码。“很多都会失败,但这没关系,这就是探索的一部分。”

而 Martin Fowler 则补充了他对技术浪潮的“二阶思考”:

“你必须在‘怀疑主义’和‘好奇心’之间找到完美的平衡。我对区块链就极其怀疑。但我的怀疑主义必须是绝对的——这意味着,我必须连我自己的怀疑本身,都保持怀疑。”

他坦言,自己一开始对 Copilot 这种东西也极度不屑,觉得它生成的都是垃圾。直到他读了 Simon Willison 的博客,才意识到:要用好一个工具,你必须先学会如何用好它。这和当年很多人嘲笑“面向对象”没用,但其实只是他们自己没有用对,是同一个道理。

戳破幻觉:“敏捷”的敌人,从来不是瀑布开发

当被问及“AI 承诺的‘更快、更好、更便宜’,与 25 年前敏捷宣言的初衷是否一致”时,Kent Beck 抛出了一个极其扎心的观点:

“事实证明,企业根本不想要更快、更好、更便宜。在一个公司内部,各种激励机制的错位,导致他们会惩罚那些真正追求效率的人。”

Martin Fowler 对此深有同感。他认为,AI 与敏捷最大的不同在于,当年他们需要费尽口舌去说服企业“敏捷有多重要”,而今天,没有任何一家公司敢对 AI 的重要性视而不见。

但这恰恰是最大的陷阱。

当年的“敏捷转型”,在无数企业中最终都演变成了一场“形式主义的灾难”,催生了庞大的“敏捷工业复合体”。

而今天,同样的剧本正在 AI 身上重演。无数根本不懂技术的咨询公司,正在兜售着各种“AI 转型”的灵丹妙药。

AI 正在成为新的“蛇油(Snake Oil)”。

注:“蛇油”是19 世纪的美国民间骗局,有人贩卖一种据说能治百病的“蛇油”之类的神药。其核心特征是用夸张的疗效宣传、用故事/神秘疗法包装、同时缺乏科学依据,最后你花钱买到的往往是没用甚至有害的东西。

架构师的终极拷问:AI 正在摧毁程序员的“社交”

如果说对“蛇油”的警惕还只是宏观层面的担忧,那么 Kent Beck 接下来提出的观点,则直接刺向了每一个正在享受 AI 编码便利的开发者。

他认为,AI 正在让软件开发“重新孤岛化(Re-soloing of programming)”

“极限编程(XP)很大一部分工作,是为那些天生不善社交的程序员,创造一个安全的社交环境。在一个 XP 团队里,人们每天花几个小时进行结对编程、激烈讨论,并乐在其中。”

“但我现在看到的是什么?‘我是一个程序员,我手下有 6 个 Agent,所以我是一个小团队的管理者。’ 不,你不是。你只是在同时使用 6 个工具。

在过去,我们把程序员从一个个封闭的办公室里解放出来,让他们围坐在一起,通过“混乱、复杂、充满人味儿”的社交过程,去创造伟大的软件。

而现在,我们似乎又在主动退回那个“把程序员关进小黑屋,从门缝底下塞披萨”的时代。只不过,这次陪伴你的,是几个冰冷的 AI 机器人。

Martin Fowler 也表达了同样的担忧:

“未来的团队,到底是‘一个披萨的团队’(因为 Agent 不吃披萨),还是一个‘两个披萨的团队,但效率翻倍’?我赌后者。

他认为,“两个人类 + N 个 AI” 的结对编程模式,可能是未来的答案。因为两个人类可以更好地控制 AI 的方向,同时保留了宝贵的人类交互。

有趣的是,Kent Beck 甚至觉得现在的 AI 有点“太快了”。

“当 AI 需要 3 分钟才能返回结果时,我们正好可以利用这段时间,去讨论一下变量命名的哲学,或者下一步的架构方向。但如果它 15 秒就返回了,我们就失去了交流的时间。”

手艺人的黄昏:当 AI 剥夺了“重构的快感”

在对话的最后,当被问及“AI 时代,程序员该如何自处”时,Kent Beck 的一段独白,充满了“手艺人”的失落与悲情,足以让每一个热爱编码的资深开发者瞬间破防。

“我过去在编程中获得的一种‘强迫症’般的享受,正在消失。那种把一个文件从一坨屎山,通过无数个微小、安全的步骤,最终重构成一件艺术品的快感,再也没有了。”

“我依然可以从宏观上理解我正在做什么。但我需要把我的关注点,从享受‘雕琢程序本身’,转移到享受‘理解业务领域’上。因为在‘雕琢程序’这件事上,我们已经失去了杠杆。”

Martin Fowler 则给出了更具操作性的建议:

“一个有趣的现象是:开发者体验(Developer Experience)和智能体体验(Agent Experience)的维恩图,是一个完美的圆。对 Agent 友好的代码,对人类也友好。”

他认为,拥有良好模块化、清晰接口和完备测试的代码,AI 处理起来会更得心应手。我们过去几十年积累的那些“手艺”,并没有过时,它们只是从“指导人类”变成了“指导 AI”。

小结:在不确定的浪潮中,抓住不变的礁石

这场持续了一个多小时的对话,没有给出任何关于“如何写 Prompt”、“用哪个模型”的答案。

但这两位穿越了数个技术周期的智者,用他们的人生经验,为我们指明了在 AI 这场史无前例的巨浪中,唯一能抓住的几块礁石:

  1. 保持绝对的怀疑,包括对怀疑本身的怀疑。
  2. 学会设计最小化的实验,亲自去验证那些天花乱坠的说法。
  3. 不要放弃与人交流,那才是创造力的真正源泉。
  4. 把你的代码写得更清晰、更模块化、测试更完备。这不仅是为了你自己,更是为了你未来的 AI 同事。

最后,Kent Beck 给出了一个极其悲壮的建议:或许,我们是时候放弃享受“雕琢代码”的乐趣,而去享受“理解世界”的乐趣了。

这或许是对 AI 时代,我们这些“数字手艺人”最深刻、也最无奈的宿命注解。

资料链接:https://www.youtube.com/watch?v=CZs8J1ZD0CE


今日互动探讨:

在使用 AI 编程后,你是否也像 Kent Beck 一样,感觉失去了那种“重构屎山”的快感?在 AI 时代,你认为“结对编程”是会消亡,还是会变得更加重要?

欢迎在评论区分享你的看法!


还在为“复制粘贴喂AI”而烦恼?我的新专栏 AI原生开发工作流实战 将带你:

  • 告别低效,重塑开发范式
  • 驾驭AI Agent(Claude Code),实现工作流自动化
  • 从“AI使用者”进化为规范驱动开发的“工作流指挥家”

扫描下方二维码,开启你的AI原生开发之旅。


原「Gopher部落」已重装升级为「Go & AI 精进营」知识星球,快来加入星球,开启你的技术跃迁之旅吧!

我们致力于打造一个高品质的 Go 语言深度学习AI 应用探索 平台。在这里,你将获得:

  • 体系化 Go 核心进阶内容: 深入「Go原理课」、「Go进阶课」、「Go避坑课」等独家深度专栏,夯实你的 Go 内功。
  • 前沿 Go+AI 实战赋能: 紧跟时代步伐,学习「Go+AI应用实战」、「Agent开发实战课」、「Agentic软件工程课」、「Claude Code开发工作流实战课」、「OpenClaw实战分享」等,掌握 AI 时代新技能。
  • 星主 Tony Bai 亲自答疑: 遇到难题?星主第一时间为你深度解析,扫清学习障碍。
  • 高活跃 Gopher 交流圈: 与众多优秀 Gopher 分享心得、讨论技术,碰撞思想火花。
  • 独家资源与内容首发: 技术文章、课程更新、精选资源,第一时间触达。

衷心希望「Go & AI 精进营」能成为你学习、进步、交流的港湾。让我们在此相聚,享受技术精进的快乐!欢迎你的加入!

img{512x368}


商务合作方式:撰稿、出书、培训、在线课程、合伙创业、咨询、广告合作。如有需求,请扫描下方公众号二维码,与我私信联系。

如发现本站页面被黑,比如:挂载广告、挖矿等恶意代码,请朋友们及时联系我。十分感谢! Go语言第一课 Go语言进阶课 AI原生开发工作流实战 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