标签 Go语言 下的文章

HashiCorp 创始人亲口“认错”:AI 让我重新爱上了 Go (文末福利)

本文永久链接 – https://tonybai.com/2026/04/23/hashicorp-founder-admits-go-is-alive-thanks-to-ai

大家好,我是Tony Bai。

今天是世界读书日,在同款公众号文章的文末我将送出两个价值 99 元的《从 0 开始构建 Agent Harness》专栏的免费兑换码,欢迎大家点击这里积极留言参与!

在技术圈的江湖里,总有那么几位“扫地僧”级别的人物。他们的一言一行,足以引发整个行业的地震。Mitchell Hashimoto,正是其中之一。

作为 HashiCorp 的创始人,曾连续12年,一手使用Go 缔造了Consul、Nomad、Terraform、Vagrant、Vault 等一系列云原生基础设施与Devops“神器”以及Ghostty Terminal (使用 Zig )的他,被无数开发者奉为“云基础设施时代教父级的人物”。

但在 Go 社区,Mitchell 的形象却颇具争议。因为他曾在公开场合不止一次地表达过对 Go 语言的失望,甚至抛出过“Go has no place anymore”(Go 已无立足之地)这样的“暴论”。

然而,就在最近,这位曾经的“Go 社区的争议人物”,却在 X 平台上发表了一篇 180 度大转弯的“认错”长推,瞬间引爆了整个技术圈,获得了超过 21 万的阅读量。

他写道:

“我又开始写 Go 了……‘等等,我以为你说过 Go 已经没有位置了?’我错了。

“我错的原因,主要是因为 AI 智能体(Agent)在 Go 语言上的生产力高得惊人。我不会把其他语言扯进来,因为我不想喂饱那些螃蟹(暗指 Rust 社区)。”

是什么,让这位顶级大神发生了如此戏剧性的转变?

今天,我们就来深度扒开 Mitchell 的这篇“忏悔录”,看看在 AI Agent 席卷一切的时代,Go 语言那些曾被我们疯狂吐槽的“缺点”,是如何摇身一变,成为最顶级的“超能力”的。

惊天反转:“糟糕的人体工程学”,竟是完美的“机器工程学”

Mitchell 在推文中,首先就点出了一个极其“讽刺”的现象:

“这很有趣,因为 Go 的很多 CLI 工具,比如 go doc 和 gopls,它们糟糕的人体工程学(shitty ergonomics)……竟然被 Agent 完美地规避了。不仅如此,讽刺的是,它们对 Agent 来说简直是天赐之物。”

这句话,堪称整篇推文的点睛之笔。

如果你是一个有经验的 Go 开发者,你一定吐槽过 go doc 的简陋,或者早期 gopls 的各种不智能。相比于 Rust 的 rust-analyzer 那种极其强大的类型推断和代码补全,Go 的工具链显得既“笨”又“直白”。

但在 AI Agent 的世界里,这种“笨拙”,恰恰成了最顶级的优点!

Mitchell 指出,他现在根本不需要给 Agent 写任何复杂的 Skill。只需要在 AGENTS.md 里写一句极其简单的指令:“想找 API 或者调用者?去用 gopls。”

Agent 就能利用 gopls 提供的底层 LSP(语言服务器协议)接口,以极低的 Token 成本,精准地找到接口的实现、方法的定义,以及所有的调用关系。

另一位开发者在评论中也补充道:

“我们一直抱怨 Go 的啰嗦(verbosity),结果证明这恰恰是 LLM 最喜欢的。它们能清晰地读懂意图,而且工具链(如 go doc)给了它们足够的上下文,让它们第一次就能写出能跑通的代码。”

看懂了吗?

那些曾经被人类程序员嫌弃的“机器友好”的接口,在 AI Agent 这个“硅基程序员”面前,摇身一变成了最高效、最廉价的沟通方式。

我们过去追求的“CLI 人体工程学”,在 AI 时代,正在被“Agent 机器工程学”所降维打击。

王者归来:当“无聊”成为 AI 的最佳温床

Mitchell 的“认错”,不仅仅是因为工具链的意外适配。更深层次的原因,在于 Go 语言本身的“无聊”哲学。

在另一场由 OpenAI 创始人引发的“Go vs Rust”论战中,我们已经探讨过这个观点:

Go 语言极简的语法、强制的 gofmt 格式化、以及“万物皆 for 循环”的单一表达方式,使得所有 Go 代码库看起来都像是一个模子里刻出来的。

这种极度的“同质化”,对于基于概率预测的 AI Agent 来说,简直就是天堂。

AI 在生成 Go 代码时,不需要去猜测这个项目是函数式风格还是面向对象风格,不需要去处理复杂的生命周期和所有权问题。它只需要遵循那套刻在骨子里的“Go Way”,就能生成出八九不离十的、能跑通的代码。

评论区里,HashiCorp 的前同事现身说法:

“我当年就是看到 HashiCorp 在用 Go 才入坑的。你今天的这篇帖子,完美地解释了为什么我最近又回到了 Go 的怀抱。”

简单、可预测、没有魔法。 这些在人类极客眼中可能是“缺点”的特质,在 AI Agent 眼里,却成了最宝贵的“确定性”。

终极答案:Go + Zig,基础设施的“黄金搭档”

当然,Mitchell 也并非无脑吹捧 Go。作为一个顶级的开发者,他清醒地认识到 Go 的边界。

当他需要编写一个“可移植的、能轻松嵌入各种生态系统”的底层库时,他并没有选择 Go,而是选择了 Zig

“对我来说,重要的是可移植性。我正在写一个必须能轻松嵌入各种生态系统的通用库。一个独立的、不依赖 libc、没有操作系统原语要求、能说 C ABI、并且只有 100KB 大小的库,是一个很容易推销的方案。”

在这里,Mitchell 亮出了他的答案:Go + Zig。

  • Go:负责上层的、高并发的业务逻辑和网络调度。
  • Zig:负责底层的、需要极致性能、零依赖、跨平台 C ABI 兼容的核心组件。
  • CGO:通过 Zig 强大的交叉编译能力,将 Go 与底层 C-ABI 组件的胶水成本降到最低。

这套组合拳,既享受了 Go 无与伦比的开发效率和并发模型,又利用了 Zig 对底层的极致压榨能力,同时还避开了原生 CGO 的种种编译噩梦。

这或许是比“Go vs Rust”之争,更具前瞻性和实操价值的“版本答案”。

英雄所见略同:Pandas 之父的“痛苦告别”

如果说 Mitchell Hashimoto 的“回归”还带有一丝 云原生以及DevOps 创始人的恋旧情结,那么另一位顶级大神——Pandas 库的创始人、数据科学界的“教父级”人物 Wes McKinney——的2026表态,则更像是一封写给 Python 的“分手信”,充满了痛苦、不舍,但又极其决绝。

就在 Mitchell 的推文引发热议的同时,有人在评论区挖出了 Wes McKinney 今年年初的一篇极具前瞻性的博文《从人类工程学到智能体工程学》。

在这篇文章里,Wes McKinney 抛出了一个极其震撼的开场白:

“我最近用 Go 写了很多新软件。但问题是,我这辈子其实一行 Go 代码都没写过。这到底是怎么回事?”

答案,同样是 AI Agent。

作为一个将毕生心血都奉献给了 Python 数据科学生态的巨匠,Wes McKinney 坦言,当软件的“主要作者”从人类变成 AI 时,我们评判一门编程语言优劣的标准,发生了根本性的改变。

“人类工程学(Human Ergonomics)的重要性正在急剧下降。Python 对人类来说极其愉快和高效,但当 Agent 替你写所有代码时,这个好处就显得无足轻重了。”

他用一种近乎“残忍”的视角,剖析了 Python 在 AI Agent 时代的三个致命缺陷:

  1. 缓慢的编译-测试循环:Agent 编译和测试的频率比人类高出一到两个数量级。Python 缓慢的测试启动和依赖安装,对 Agent 来说是一种“惩罚”。
  2. 痛苦的软件分发:Agent 需要大量自包含的、无依赖的二进制工具。而 Python 拖着一个沉重的解释器,感觉就像“我们当年拼命想摆脱的 Java 虚拟机(JVM)”。
  3. 性能与内存的短板:这些在人类开发时可以容忍的问题,在 Agent 24 小时高强度运行时,会被无限放大。


图 Python Environment https://xkcd.com/1987/

那么,AI Agent 时代的“赢家”是谁?

Wes McKinney 给出了和 Mitchell Hashimoto 几乎一模一样的答案:Go。当然在数据科学以及人工智能的基础设施层面,Wes McKinney认为 Rust 也将会占据着越来越重要的地位。

因为它们解决了最关键的三个问题:

  • 无痛构建静态二进制文件。
  • 极速、确定性的构建过程。
  • 精简的资源占用和出色的运行时性能。

他甚至更进一步指出,由于 Go 拥有比 Rust 快得多的编译时间,在 Agent 高频迭代的场景下,Go 甚至比 Rust 更具优势。

“我依然深爱着 Python,并为我们建立的生态系统感到自豪。但很明显,鉴于 Agent 循环带来的生产力优势,我和业界的大部分人,将会写越来越少的 Python,转而拥抱 Go 和其他现代编译语言。”

一个为 Python 奋斗了近 20 年的灵魂人物,最终为了 AI,选择了自己从未写过的 Go。

这已经不是简单的技术选型,这是一场关于“工程师生存法则”的深刻变革。

英雄惜英雄:一场关于“回归”的集体狂欢

Mitchell 的这篇“认错”长文,像一声号角,引来了无数在 Go 与其他语言之间摇摆的开发者的共鸣。

Bun 的创始人 Jarred Sumner 激动地在评论区留言:“我想看看你到底在搞什么!”(Mitchell 回复:“我早点联系你!”)

一位前 Vercel 工程师更是直言:“老哥你终于兜了一圈又回来了!”

当然也有一些开发者表示这也许是Mitchell的“幻觉”或“偏见”,一位开发者(显然不是很熟悉 Mitchell 的开发过往)写道:

“也许你只是比 Zig 更不习惯 Go,所以你注意到的 Go 的问题更少。而且你已经是 Zig 的专家了,用它提升的空间不大了(想学习一下新的编程语言)。LLM 让你看到在你不懂的领域(指Go)正确率是 100%,但在你懂 60% 的领域(指Zig),只对了 60%”。
(Mitchell 则毫不客气地回怼:“我写了 12 年全职的、纯粹的 Go。我的判断力很可靠。”)

这场大讨论,最终演变成了一场关于“回归 Go”的集体狂欢。

小结:在 AI 时代,重新审视“简单”的价值

Mitchell Hashimoto 的故事,是 AI 时代软件工程演进的一个完美缩影。

一个曾经因为 Go 的“不够底层”、“人体工程学差”而选择离开的顶级大神,最终又因为 AI Agent 的出现,重新发现了这门语言在“机器工程学”上的巨大价值。

这提醒我们所有技术人:对一门语言的评判,永远不能脱离其所处的时代背景和生产力工具。

在人类手搓代码的时代,我们追求的是表达力的丰富和语法的灵巧。

而在 AI 自动生成的时代,简单、可预测、无歧义、易于机器理解,反而成了最稀缺的“黄金法则”。

Go 语言的缔造者们,在十几年前就用近乎偏执的克制,为我们埋下了一颗时间的种子。

直到今天,在 AI 的催化下,这颗种子,终于长成了参天大树。

资料链接:

  • https://x.com/mitchellh/status/2046319366489407803
  • https://wesmckinney.com/blog/agent-ergonomics/

今日互动探讨:

在 AI 编程的浪潮中,你是否也像 Mitchell 一样,重新审视了自己对某门语言的看法?你认为在 AI Agent 眼里,最“友好”和最“劝退”的语言分别是什么?

欢迎在评论区分享你的观点!


世界图书日特别福利:一本定义未来的“活书”

今天(4月23日)就是世界图书日。

在这个属于知识与智慧的节日里,与其被动地阅读别人写的书,不如我们亲手来“写”一本定义未来的“书”——构建一个属于你自己的 AI Agent Harness

Mitchell Hashimoto 和 Wes McKinney 的故事告诉我们,AI Agent 正在成为这个时代最强大的生产力杠杆。而驾驭这头巨兽的核心,不在于你会背多少 Prompt,而在于你是否懂得如何为它构建一个坚不可摧的“驾驭系统(Harness)”。

为了庆祝我的全新极客时间专栏 从 0 开始构建 Agent Harness 上线,并感谢大家一直以来的支持,我将拿出 2 个免费的专栏兑换码送给大家!

参与方式:

关注本公众号,并在本文的评论区留言,聊一聊:“在 AI Agent 时代,你认为一个程序员最不可被替代的核心技能是什么?为什么?”

我将在 72 小时后,从所有精选留言中,挑选 2 位最深刻、最走心的思考,每人赠送一份价值 99 元的《从 0 开始构建 Agent Harness》专栏兑换码。


还在为写 Agent 框架频频死循环、上下文爆炸而束手无策?我的新专栏 从0 开始构建 Agent Harness 将带你:

  • 抛弃臃肿框架,回归“驾驭工程 (Harness Engineering)”的第一性原理
  • 用 Go 语言手写 ReAct 循环、并发拦截与上下文压缩引擎等,复刻极简OpenClaw
  • 构建坚不可摧的 Safety Middleware 与飞书人工审批防线
  • 在底层实现 Token 成本审计、链路追踪与自动化跑分评估
  • 从“调包侠”进化为掌控大模型边界的“AI 操作系统架构师”

扫描下方二维码,开启从 0 开始构建Agent Harness 的实战之旅。


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

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

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


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

为什么说 go 语句是新时代的 goto?四大法则拯救失控 goroutine

本文永久链接 – https://tonybai.com/2026/04/16/structured-concurrency-in-go-research-oriented-perspective

大家好,我是Tony Bai。

Go 语言的 go 关键字是并发编程史上的一次民主化革命,它让并发变得前所未有的廉价和简单。只需在一个函数调用前加上 go,我们就拥有了一个并发执行的任务。

这种语法是如此的诱人,以至于新手 Gopher 往往会沉迷于创建成千上万个 Goroutine。

随着 Go 语言步入第 16 个年头,学术界和工程界也开始重新审视这种“极简主义”带来的副作用。

2025 年 3 月,一篇发表在《Scientific Research Journal》上的重磅论文《Structured Concurrency in Go: A Research-Oriented Perspective》,将 Go 的并发模型与 1968 年 Dijkstra 对 Goto 语句的批判联系了起来。

论文作者 Georgii Kliukovkin 指出,这种“发射后不管(Fire-and-Forget)”的模式,虽然在 Hello World 级别的程序中运行良好,但在大规模分布式系统中,它是资源泄漏、死锁和竞态条件的温床。

我们日常也常听到这样的抱怨:“Go 的并发很简单,但写出正确的并发代码很难。” 这并非语言本身的缺陷,而是因为我们缺乏一种与语言灵活性相匹配的约束纪律。这种纪律,就是结构化并发

本文将深入解读这篇论文,探讨为何“不受限制的 Goroutine”正在成为新时代的“Goto 语句”,以及我们如何通过结构化并发(Structured Concurrency)的四大法则,将失控的协程重新关回笼子,构建坚如磐石的系统。

历史的镜像——从 Goto 有害论到 Goroutine 有害论?

要理解“结构化并发”,我们必须先回顾历史。

1968年的呼喊:结构化编程的诞生

在 20 世纪 60 年代,编程界流行的是“非结构化编程”。开发者可以随心所欲地使用 goto 语句在代码的任意位置跳转。这种自由带来了极大的灵活性,但也导致了所谓的“意大利面条代码(Spaghetti Code)”——控制流杂乱无章,难以追踪程序的执行路径,维护简直是噩梦。

1968 年,图灵奖得主 Edsger W. Dijkstra 发表了那篇著名的《Go To Statement Considered Harmful》(Goto 语句有害论)。他主张废除无限制的跳转,转而使用结构化编程(Structured Programming):即所有的逻辑都应由顺序结构、选择结构(if/else)和循环结构(for/while)以及函数调用(Function Call)组成。

结构化编程的核心价值在于“黑盒化”。当你调用一个函数时,你确信控制权最终会回到你手中(除非死循环或崩溃);你确信该函数内部的变量不会污染外部环境。这种“入口-出口”的对称性,是软件可维护性的基石。

2025年的回响:go 语句 即 Goto

论文提出了一个让人振聋发聩的观点:Go 语言中的 go 语句,在某种意义上,就是并发领域的 goto。

当你执行 go func() 时,你实际上是启动了一个新的执行流,它跳出了当前的词法作用域(Lexical Scope)。

  • 它什么时候开始?不确定。
  • 它什么时候结束?不知道。
  • 它如果 Panic 了会怎样?可能会炸掉整个程序。
  • 父函数返回了,它还在运行吗?很有可能。

这种“射后不理(Fire-and-Forget)”的模式,破坏了代码的封装性。就像当年的 goto 打破了控制流的结构一样,不受约束的 go 语句打破了并发流的结构。

结构化并发的目标,就是要把这些“野生”的 Goroutine 重新关进“代码块”的笼子里,让并发程序的生命周期像同步程序一样清晰、可预测。

打破幻象——Go 并发的三个误区

在引入解决方案之前,论文首先抨击了 Go 社区中常见的三个关于并发的迷思。这些误区往往是导致系统不稳定的根源。

误区 1:“Goroutine 极度廉价,所以可以随便开”

是的,Goroutine 的初始栈只有 2KB,但这只是“内存”成本。从“生命周期”的角度看,一个泄露的 Goroutine 是极其昂贵的。

如果不加控制地启动 Goroutine 而不确保其退出,这些“孤儿”协程可能会:

  • 持有数据库连接或文件句柄不释放。
  • 阻塞在某个永远不会发送数据的 Channel 上。
  • 阻止垃圾回收器(GC)回收其引用的对象。

在长期运行的服务中,这种微小的泄漏会像滚雪球一样,最终导致服务 OOM(内存溢出)。

误区 2:“Channel 解决了所有同步问题”

Rob Pike 的名言“不要通过共享内存来通信,要通过通信来共享内存”被许多人奉为圭臬。然而,Channel 并不是银弹。

Channel 实际上引入了复杂的状态机问题:

  • 向已关闭的 Channel 发送数据会 Panic。
  • 从 nil Channel 读取会永久阻塞。
  • 无缓冲 Channel 容易导致死锁。
  • 过多的 Channel 会导致逻辑碎片化,增加认知负担。

论文强调,Channel 是一种传输机制,而不是一种架构保障。没有设计良好的生命周期管理,Channel 只会让 Bug 变得更难调试。

误区 3:“Go 的并发代码很容易测试”

Go 提供了 go test -race,但这远远不够。并发 Bug 往往是非确定性的(Heisenbugs),在本地开发环境(低负载、少核)下可能永远不会出现,一上生产环境(高负载、多核)就崩溃。

如果代码缺乏结构化,测试将变得极其困难。你无法确定在断言(Assert)的那一刻,后台的 Goroutine 是否已经完成了数据的写入。结构化并发通过明确的“等待”机制,能让并发测试变得像同步测试一样稳定。

核心法则——构建坚固的并发大厦

既然 Go 语言层面(目前)没有强制的结构化并发语法(不同于 Java Project Loom 的 StructuredTaskScope 或 Python Trio 的 Nursery),我们需要依靠工程纪律和设计模式来实现它。论文详细阐述了四大核心法则。

法则一:Scope 闭环原则 —— 在谁的 Scope 启动,就在谁的 Scope 等待

定义任何启动 Goroutine 的函数,必须负责等待它们结束。

这是结构化并发的第一天条。绝不允许 Goroutine 的生命周期“逃逸”出启动它的函数。这保证了当函数返回时,它所衍生的所有并发工作都已完结,资源已释放。

❌ 反模式:泄露的抽象

// 这是一个危险的模式:函数返回了,但后台任务还在跑
// 调用者无法知道任务何时完成,也无法处理 panic
func FireAndForget() {
    go func() {
        // 执行一些可能会阻塞很久的任务
        // 这里发生的一切,父函数都无法控制
    }()
}

✅ 正模式:Wait 优于 Sleep

论文强烈建议使用 sync.WaitGroup 或 errgroup 来显式地界定生命周期边界。

func ProcessStructured(items []Data) {
    var wg sync.WaitGroup

    for _, item := range items {
        wg.Add(1)
        // 使用闭包捕获变量时需注意
        go func(val Data) {
            defer wg.Done()
            process(val)
        }(item)
    }

    // 关键点:在函数返回前,必须收敛所有并发流
    // 这形成了一个清晰的“并发块”
    wg.Wait()
}

通过这种方式,ProcessStructured 函数的行为变成了“同步”的黑盒。调用者不需要知道它内部是否使用了并发,只需要知道“当函数返回时,所有工作都已完成”。

法则二:同步外观原则 —— API 应当表现为“同步”

定义即使函数内部使用了高并发,对外暴露的 API 签名应当是同步阻塞的。

这是一个看似反直觉的建议。既然我们写的是并发程序,为什么 API 要设计成同步的?

论文指出,异步 API(如返回一个 <-chan Result 或 Future)具有“传染性”。一旦你的函数返回了一个 Future,调用者就必须处理这个 Future 的等待逻辑,这会层层向上传递,导致整个调用链都充满了并发管理的细节。

经典案例:http.ListenAndServe

Go 标准库的 http.ListenAndServe(“:8080″, nil) 是结构化并发 API 设计的典范。

  • 内部:它是一个极其复杂的并发系统,为每个进来的 TCP 连接启动一个新的 Goroutine。
  • 外部:它是一个简单的阻塞函数。
// 调用者代码
err := http.ListenAndServe(":8080", nil)

// 当这行代码返回时,我们确切地知道:
// 1. 服务已经停止了。
// 2. 或者发生了错误(如端口冲突)。

如果 ListenAndServe 被设计成异步返回(即在后台启动服务后立即返回),那么调用者将面临巨大的困扰:我该如何知道服务启动成功了?如果启动失败,错误去哪里了?主进程该何时退出?

除非是专门的任务调度器,否则业务逻辑函数的 API 应该看起来是同步阻塞的。让调用者去决定是否使用 go 关键字来调用它。

法则三:所有权原则 —— 在哪写入,就在哪关闭

定义只有负责向 Channel 写入数据的 Goroutine,才有资格关闭该 Channel。

Channel 的关闭操作是 Go 并发中最容易导致 Panic 的环节(向已关闭的 Channel 发送数据)。论文强调,结构化并发可以极大地简化 Channel 的管理。

原则非常简单:谁生产,谁负责清理。 接收者(Consumer)永远不应该关闭 Channel,因为通过关闭 Channel 来通知生产者“我读完了”是一种错误的设计(应该使用 Context 来取消)。

结合法则一,如果生产者 Goroutine 的生命周期是受控的,那么 Channel 的生命周期自然也是受控的。

func Producer() <-chan int {
    ch := make(chan int)

    // 启动生产者协程
    go func() {
        // defer close 确保无论正常退出还是 panic,channel 都会关闭
        // 避免接收者永久阻塞
        defer close(ch) 

        for i := 0; i < 10; i++ {
            ch <- i
        }
    }()

    return ch
}

法则四:物理封装原则 —— 数据与锁不分家

定义将共享的可变数据(Mutable State)与保护它的同步原语(Mutex)封装在同一个结构体中。

在共享内存的并发模型中,最大的噩梦是“锁与数据分离”。例如,你定义了一个全局变量 var Cache map[string]int,然后又定义了一个全局锁 var Mu sync.Mutex。随着代码量的增加,开发者很容易忘记在访问 Cache 时加锁,或者错误地使用了其他的锁。

论文建议采用一种“物理强绑定”的策略:

type SafeCounter struct {
    // 1. 将锁作为结构体的第一个字段
    mu sync.Mutex

    // 2. 受保护的数据应当是私有的(小写)
    // 强制外部必须通过方法来访问
    values map[string]int
}

// 3. 只有通过这个方法才能访问数据
func (c *SafeCounter) Inc(key string) {
    c.mu.Lock()
    // 4. 利用 defer 确保锁的释放与函数作用域绑定
    defer c.mu.Unlock()

    c.values[key]++
}

这种模式被称为 Monitor Pattern(监视器模式)。它通过封装强制实施了并发安全,将“会不会加锁”的问题变成了“能不能调用方法”的问题,后者由编译器保证,前者只能靠人品。

进阶——超越标准库的尝试

虽然标准库提供了 sync.WaitGroup 和 context,但要完美实现结构化并发,样板代码依然繁多。论文提到了社区中一些优秀的尝试,其中最值得关注的是 Sourcegraph 开源的 conc 库

conc 库试图解决标准库 WaitGroup 的两个痛点:

  1. Panic 逃逸:在标准 go func 中,如果子协程 panic,整个程序会直接崩溃(Crash),父协程无法 recover。这对于高可用服务是致命的。
  2. Error 传播:WaitGroup 不支持错误返回,需要开发者自己维护一个 err 变量或使用 errgroup。

conc 提供了增强版的 WaitGroup:

import "github.com/sourcegraph/conc"

func main() {
    var wg conc.WaitGroup

    wg.Go(func() {
        // 如果这里 panic 了
        panic("something went wrong")
    })

    // Wait() 会自动捕获子协程的 panic
    // 并将其重新抛出或作为错误返回(取决于具体 API)
    // 从而避免进程直接崩溃
    wg.Wait()
}

这种工具库的出现,标志着 Go 社区正在从“手动管理并发”向“自动化管理并发”演进,这正是结构化并发理念的工程化落地。

小结:从“能用”到“可控”

Go 语言通过 go 关键字将并发编程的门槛降到了历史最低,赢得了云计算时代的入场券。但在构建大规模、高可靠的系统时,我们不能止步于“能用”。

这篇学术论文为我们提供了一个冷静的视角:并发不是目的,只是手段。 失控的并发是灾难,只有受控的并发才是生产力。

结构化并发不是一种束缚,而是一种保护。它要求我们在写下每一个 go func 的时候,都要问自己三个问题:

  1. 它什么时候结束?
  2. 谁负责等待它结束?
  3. 如果它出错了,谁来处理?

只有当这三个问题都有明确答案时,我们才能说,我们真正掌握了 Go 的并发艺术。

参考资料


你更倾向于哪一派?

有人认为 Go 的自由是生产力之源,有人认为约束才是工程的救赎。在你的项目中,你是否也曾因为“射后不理”的 goroutine 踩过坑?你认为 Go 官方是否应该在语言层面引入类似 Java 或 Python 的结构化并发原生支持?

欢迎在评论区分享你的看法或“血泪史”!

想深入掌握 Go 并发调度的底层原理?点击查看我的微专栏《Go 并发调度艺术》。


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

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

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


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

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

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

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

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


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

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