<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0"
	xmlns:content="http://purl.org/rss/1.0/modules/content/"
	xmlns:wfw="http://wellformedweb.org/CommentAPI/"
	xmlns:dc="http://purl.org/dc/elements/1.1/"
	xmlns:atom="http://www.w3.org/2005/Atom"
	xmlns:sy="http://purl.org/rss/1.0/modules/syndication/"
	xmlns:slash="http://purl.org/rss/1.0/modules/slash/"
	>

<channel>
	<title>Tony Bai &#187; 编译器</title>
	<atom:link href="http://tonybai.com/tag/%e7%bc%96%e8%af%91%e5%99%a8/feed/" rel="self" type="application/rss+xml" />
	<link>https://tonybai.com</link>
	<description>一个程序员的心路历程</description>
	<lastBuildDate>Sun, 12 Apr 2026 22:30:28 +0000</lastBuildDate>
	<language>en</language>
	<sy:updatePeriod>hourly</sy:updatePeriod>
	<sy:updateFrequency>1</sy:updateFrequency>
	<generator>http://wordpress.org/?v=3.2.1</generator>
		<item>
		<title>Rust 看了流泪，AI 看了沉默：扒开 Go 泛型最让你抓狂的“残疾”类型推断</title>
		<link>https://tonybai.com/2026/03/27/function-type-inference-should-work-in-all-assignment-contexts/</link>
		<comments>https://tonybai.com/2026/03/27/function-type-inference-should-work-in-all-assignment-contexts/#comments</comments>
		<pubDate>Thu, 26 Mar 2026 23:09:11 +0000</pubDate>
		<dc:creator>bigwhite</dc:creator>
				<category><![CDATA[技术志]]></category>
		<category><![CDATA[AIProgramming]]></category>
		<category><![CDATA[AI编程]]></category>
		<category><![CDATA[Assignability]]></category>
		<category><![CDATA[ClaudeCode]]></category>
		<category><![CDATA[Codex]]></category>
		<category><![CDATA[Compiler]]></category>
		<category><![CDATA[CompositeLiterals]]></category>
		<category><![CDATA[ErrorHandling]]></category>
		<category><![CDATA[generics]]></category>
		<category><![CDATA[Go]]></category>
		<category><![CDATA[Golang]]></category>
		<category><![CDATA[GoLanguage]]></category>
		<category><![CDATA[Go语言]]></category>
		<category><![CDATA[Instantiation]]></category>
		<category><![CDATA[Issue77245]]></category>
		<category><![CDATA[RobertGriesemer]]></category>
		<category><![CDATA[SoftwareEngineering]]></category>
		<category><![CDATA[StaticTyping]]></category>
		<category><![CDATA[SyntacticSugar]]></category>
		<category><![CDATA[typeinference]]></category>
		<category><![CDATA[TypeSystem]]></category>
		<category><![CDATA[可赋值性]]></category>
		<category><![CDATA[复合字面量]]></category>
		<category><![CDATA[实例化]]></category>
		<category><![CDATA[泛型]]></category>
		<category><![CDATA[类型推断]]></category>
		<category><![CDATA[类型系统]]></category>
		<category><![CDATA[编译器]]></category>
		<category><![CDATA[语法糖]]></category>
		<category><![CDATA[软件工程]]></category>
		<category><![CDATA[错误处理]]></category>
		<category><![CDATA[静态类型]]></category>

		<guid isPermaLink="false">https://tonybai.com/?p=6106</guid>
		<description><![CDATA[本文永久链接 &#8211; https://tonybai.com/2026/03/27/function-type-inference-should-work-in-all-assignment-contexts 大家好，我是Tony Bai。 在这个大模型（AI）写代码如喝水一般简单的时代，你有没有遇到过一种极其憋屈的场景： 你让 Claude Code 或者 Codex 帮你写了一段 Go 语言代码，逻辑清晰，结构优雅，连它自己都觉得这波操作满分。但当你满怀期待地按下 go run 时，Go 编译器却无情地丢给你一个红色报错： cannot use generic function g without instantiation （不能在未实例化的情况下使用泛型函数 g） AI 沉默了，它不明白自己错在哪；如果你是个习惯了 Rust 那种“地表最强类型推断”的开发者，你可能会当场流下心酸的眼泪—— 在 Rust 里闭着眼睛都能推断出来的泛型参数，怎么到了 Go 里，它就突然变成了“残疾”？ 如果你曾经被这个“诡异”的泛型报错折磨过，甚至因此怀疑过自己的智商，不要怪 AI 不懂 Go 语言。 因为就在最近，连“Go 语言之父之一” 的 Robert Griesemer 都亲自在官方 GitHub 上提了一个 Issue，承认这个语法限制不仅反直觉，甚至一度被认为是一个编译器 Bug！Griesemer 本人随即在 Issue 中自我更正，明确这需要语言规范(spec)层面的修改，而不只是修编译器。 今天，我们就来扒开这个在 [...]]]></description>
			<content:encoded><![CDATA[<p><img src="https://tonybai.com/wp-content/uploads/2026/function-type-inference-should-work-in-all-assignment-contexts-1.png" alt="" /></p>
<p><a href="https://tonybai.com/2026/03/27/function-type-inference-should-work-in-all-assignment-contexts">本文永久链接</a> &#8211; https://tonybai.com/2026/03/27/function-type-inference-should-work-in-all-assignment-contexts</p>
<p>大家好，我是Tony Bai。</p>
<p>在这个大模型（AI）写代码如喝水一般简单的时代，你有没有遇到过一种极其憋屈的场景：</p>
<p>你让 Claude Code 或者 Codex 帮你写了一段 Go 语言代码，逻辑清晰，结构优雅，连它自己都觉得这波操作满分。但当你满怀期待地按下 go run 时，Go 编译器却无情地丢给你一个红色报错：</p>
<pre><code>cannot use generic function g without instantiation
（不能在未实例化的情况下使用泛型函数 g）
</code></pre>
<p>AI 沉默了，它不明白自己错在哪；如果你是个习惯了 Rust 那种“地表最强类型推断”的开发者，你可能会当场流下心酸的眼泪—— 在 Rust 里闭着眼睛都能推断出来的泛型参数，怎么到了 Go 里，它就突然变成了“残疾”？</p>
<p>如果你曾经被这个“诡异”的泛型报错折磨过，甚至因此怀疑过自己的智商，不要怪 AI 不懂 Go 语言。</p>
<p>因为就在最近，连“Go 语言之父之一” 的 Robert Griesemer 都亲自在官方 GitHub 上提了一个 Issue，承认这个语法限制不仅反直觉，甚至一度被认为是一个编译器 Bug！Griesemer 本人随即在 Issue 中自我更正，明确这需要语言规范(spec)层面的修改，而不只是修编译器。</p>
<p>今天，我们就来扒开这个在 Go 官方仓库引发热议的 <a href="https://github.com/golang/go/issues/77245">Issue #77245</a>，看看这个即将改变Go工程师日常编码的“底层规范级修补”，到底是怎么回事。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2026/agentic-software-engineering-qr.png" alt="" /></p>
<h2>“薛定谔”式的类型推断</h2>
<p>自从 Go 1.18 引入泛型以来，“不够聪明”的类型推断（Type Inference）就一直被开发者诟病。直到 Go 1.21 发布，官方宣称大幅增强了这部分能力：<strong>只要在赋值上下文中，目标类型是明确的，Go 就可以帮你自动推断出泛型函数的参数类型，不需要你手动写 g[int] 了。</strong></p>
<p>这听起来很美好，对吧？</p>
<p>但现实是极其骨感的。我们来看看 Robert Griesemer 亲自给出的这个“薛定谔式的推断”的例子：</p>
<pre><code class="go">type S struct{ f func(int) }

func g[T any](T) {} // 这是一个简单的泛型函数

func _(s S) {
    s.f = g          // ✅ 没问题！Go 编译器智商在线，完美推断出 T 是 int

    s = S{f: g}      // ❌ 报错：不能在没有实例化的情况下使用泛型函数 g

    s = S{f: g[int]} // ✅ 没问题！必须手动写死 g[int]
}
</code></pre>
<p>看懂这个坑在哪里了吗？</p>
<p>当你写 s.f = g 的时候，编译器智商在线，它知道 s.f 需要一个 func(int)，所以它机智地把泛型函数 g 实例化成了 g[int]。</p>
<p><strong>但是（最气人的但是）！</strong></p>
<p>当你使用结构体字面量 S{f: g} 进行初始化时，编译器却突然“智力下线”了。它死活推断不出 g 需要被实例化为 int，非逼着你极其啰嗦地写上 g[int]！</p>
<p>这种“一半聪明，一半智障”的表现，不仅存在于结构体里。在切片（Slice）、数组、Map，甚至是 Channel 的发送操作中：</p>
<pre><code>type F func(int)
type A [10]F
type S []F
type M map[string]F
type C chan F

func g[T any](T) {}

func _() {
    var a A
    a[0] = g      // ok
    a = A{g}      // error: cannot use generic function g without instantiation
    a = A{g[int]} // ok

    var s S
    s[0] = g      // ok
    s = S{g}      // error: cannot use generic function g without instantiation
    s = S{g[int]} // ok

    var m M
    m["foo"] = g         // ok
    m = M{"foo": g}      // error: cannot use generic function g without instantiation
    m = M{"foo": g[int]} // ok

    var c C
    c &lt;- g      // error: cannot use generic function g without instantiation
    c &lt;- g[int] // ok
}
</code></pre>
<p>只要你使用了复合字面量（Composite Literals），这套“残疾”的类型推断就会集体失效。</p>
<h2>为什么 Rust 和 AI 看了会沉默？</h2>
<p>如果你去问一个 Rust 开发者：<em>“目标结构体的字段类型 f func(int) 明明就摆在那里，Go 编译器为什么会看不见？”</em></p>
<p>Rust 开发者可能会拍着你的肩膀叹气。在 Rust 强大的类型推断系统面前，这种上下文推导简直是基本操作，根本不需要开发者操心。</p>
<p>而在如今 AI 辅助编程大行其道的时代，这个问题更加被无限放大。</p>
<p>大模型在学习了海量代码后，它的“直觉（Next-token prediction）”告诉它，这里上下文极其明确，根本不需要写死类型参数。于是 AI 开心地生成了 S{f: g}，结果却被 Go 编译器无情打脸。你不得不停止思考，手动去把 AI 生成的代码一行行加上 [int]、[string]……</p>
<p>这根本不是 AI 的幻觉，而是 Go 语言规范（Spec）在当年设计时，由于过于严谨，给自己留下的思维盲区。</p>
<p>在最初的 Go Spec 中，关于泛型函数实例化生效的上下文规定得极其死板（只在某些直接赋值的场景生效）。当时的 Go 团队并没有抽象出一个统一的 <strong>“赋值上下文（Assignment Context）”</strong> 概念。这导致散落在各个角落的复合字面量操作，全都成了漏网之鱼。</p>
<h2>官方的修补：一场牵一发而动全身的“规范手术”</h2>
<p>起初，Robert Griesemer 以为这只是个单纯的编译器 Bug，只要改改代码就行了。</p>
<p>但随着讨论的深入，核心成员们（如 Austin Clements）发现，这事儿没那么简单。要从根本上解决这个问题，<strong>必须对 Go 语言规范（Spec）动刀子！</strong></p>
<p>在随后的内部评审中，Go 团队做出了一个决策：</p>
<p>他们没有选择“头痛医头，脚痛医脚”地去给结构体、Map、切片分别打补丁。而是选择在 Go 语言最底层的定义——<strong>“可赋值性（Assignability）”</strong> 上做文章。</p>
<p>他们提出了一个<a href="https://go.dev/cl/751312">新的 CL</a> ，只要一个表达式符合“可赋值性”的校验（无论是等号赋值、结构体初始化、还是 Channel 发送），Go 编译器就必须启动泛型函数的自动类型推断。</p>
<p>这就好比给整个 Go 语言的类型推断系统，<strong>彻底打通了奇经八脉</strong>。</p>
<h2>小结</h2>
<p>到这里，可能有开发者会问：“不就是少写几个 [int] 吗？至于这么大惊小怪吗？”</p>
<p>在几行代码的 Demo 里，这确实不是事。</p>
<p>但在大厂动辄十几万或几十万行的微服务源码中，当我们使用泛型去实现高阶的“工厂模式”、“回调注册”、“依赖注入”时，代码中会充斥着大量的结构体初始化和泛型函数传递。</p>
<p>如果没有统一的类型推断，原本极其优雅的代码，就会变成被各种中括号 [T, K, V] 塞满的“乱码”。</p>
<p><strong>更少的手动类型标记，意味着更低的人类认知负荷（Cognitive Load），以及对 AI 代码生成工具更友好的兼容性。</strong></p>
<p>Go 语言之所以能在一众花里胡哨的新语言中稳坐云原生霸主的交椅，靠的绝不仅是并发，更是这种对“代码清爽度”和“心智负担”极其克制、甚至有些偏执的追求。</p>
<p>好消息是，这个被开发者诟病已久的痛点，已经被 Go 官方提案评审委员会 <strong>“正式接受（Accepted）”</strong>。</p>
<p>我们极有可能在即将到来的后续版本(比如Go 1.27)中，看到这段啰嗦的泛型代码彻底消失。</p>
<p>资料链接：</p>
<ul>
<li>https://github.com/golang/go/issues/77245</li>
<li>https://go.dev/cl/751312</li>
</ul>
<hr />
<p><strong>今日互动探讨：</strong></p>
<p>在日常写 Go 泛型的时候，你还遇到过哪些让你觉得“Go 编译器简直是个智障”的奇葩场景？或者在对比 Rust/TS 时，你觉得 Go 的类型系统最需要补齐哪个短板？</p>
<p>欢迎在评论区疯狂吐槽与分享!</p>
<hr />
<p>还在为“复制粘贴喂AI”而烦恼？我的新专栏 <strong>《<a href="http://gk.link/a/12EPd">AI原生开发工作流实战</a>》</strong> 将带你：</p>
<ul>
<li>告别低效，重塑开发范式</li>
<li>驾驭AI Agent(Claude Code)，实现工作流自动化</li>
<li>从“AI使用者”进化为规范驱动开发的“工作流指挥家”</li>
</ul>
<p>扫描下方二维码，开启你的AI原生开发之旅。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/ai-native-dev-workflow-qr.png" alt="" /></p>
<hr />
<p><strong>原「Gopher部落」已重装升级为「Go &amp; AI 精进营」知识星球，快来加入星球，开启你的技术跃迁之旅吧！</strong></p>
<p>我们致力于打造一个高品质的 <strong>Go 语言深度学习</strong> 与 <strong>AI 应用探索</strong> 平台。在这里，你将获得：</p>
<ul>
<li><strong>体系化 Go 核心进阶内容:</strong> 深入「Go原理课」、「Go进阶课」、「Go避坑课」等独家深度专栏，夯实你的 Go 内功。</li>
<li><strong>前沿 Go+AI 实战赋能:</strong> 紧跟时代步伐，学习「Go+AI应用实战」、「Agent开发实战课」、「Agentic软件工程课」、「Claude Code开发工作流实战课」、「OpenClaw实战分享」等，掌握 AI 时代新技能。 </li>
<li><strong>星主 Tony Bai 亲自答疑:</strong> 遇到难题？星主第一时间为你深度解析，扫清学习障碍。</li>
<li><strong>高活跃 Gopher 交流圈:</strong> 与众多优秀 Gopher 分享心得、讨论技术，碰撞思想火花。</li>
<li><strong>独家资源与内容首发:</strong> 技术文章、课程更新、精选资源，第一时间触达。</li>
</ul>
<p>衷心希望「Go &amp; AI 精进营」能成为你学习、进步、交流的港湾。让我们在此相聚，享受技术精进的快乐！欢迎你的加入！</p>
<p><img src="http://image.tonybai.com/img/tonybai/gopher-and-ai-tribe-zsxq-small-card.jpg" alt="img{512x368}" /></p>
<hr />
<p>商务合作方式：撰稿、出书、培训、在线课程、合伙创业、咨询、广告合作。如有需求，请扫描下方公众号二维码，与我私信联系。</p>
<p><img src="http://image.tonybai.com/img/tonybai/iamtonybai-wechat-qr.png" alt="" /></p>
<p style='text-align:left'>&copy; 2026, <a href='https://tonybai.com'>bigwhite</a>. 版权所有. </p>
]]></content:encoded>
			<wfw:commentRss>https://tonybai.com/2026/03/27/function-type-inference-should-work-in-all-assignment-contexts/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>从第一位程序员到 AI 时代的领航者：代码世界里的“她”力量</title>
		<link>https://tonybai.com/2026/03/08/her-power-in-code-pioneers-to-ai-era/</link>
		<comments>https://tonybai.com/2026/03/08/her-power-in-code-pioneers-to-ai-era/#comments</comments>
		<pubDate>Sat, 07 Mar 2026 23:59:18 +0000</pubDate>
		<dc:creator>bigwhite</dc:creator>
				<category><![CDATA[技术志]]></category>
		<category><![CDATA[AdaLovelace]]></category>
		<category><![CDATA[AIAge]]></category>
		<category><![CDATA[AIAlignment]]></category>
		<category><![CDATA[AIEthics]]></category>
		<category><![CDATA[AI伦理]]></category>
		<category><![CDATA[AI对齐]]></category>
		<category><![CDATA[AI时代]]></category>
		<category><![CDATA[AsynchronousExecutive]]></category>
		<category><![CDATA[ClaudeCode]]></category>
		<category><![CDATA[COBOL]]></category>
		<category><![CDATA[Codex]]></category>
		<category><![CDATA[CommunicationSkills]]></category>
		<category><![CDATA[Compiler]]></category>
		<category><![CDATA[ComputerPioneers]]></category>
		<category><![CDATA[DeepLearning]]></category>
		<category><![CDATA[Diversity]]></category>
		<category><![CDATA[docker]]></category>
		<category><![CDATA[FeiFeiLi]]></category>
		<category><![CDATA[GraceHopper]]></category>
		<category><![CDATA[ImageNet]]></category>
		<category><![CDATA[InternationalWomensDay]]></category>
		<category><![CDATA[JessieFrazelle]]></category>
		<category><![CDATA[MargaretHamilton]]></category>
		<category><![CDATA[PromptEngineering]]></category>
		<category><![CDATA[SoftwareEngineering]]></category>
		<category><![CDATA[SystemsThinking]]></category>
		<category><![CDATA[国际妇女节]]></category>
		<category><![CDATA[多元化]]></category>
		<category><![CDATA[异步调度]]></category>
		<category><![CDATA[提示词工程]]></category>
		<category><![CDATA[李飞飞]]></category>
		<category><![CDATA[沟通能力]]></category>
		<category><![CDATA[深度学习]]></category>
		<category><![CDATA[系统思维]]></category>
		<category><![CDATA[编译器]]></category>
		<category><![CDATA[计算机先驱]]></category>
		<category><![CDATA[软件工程]]></category>

		<guid isPermaLink="false">https://tonybai.com/?p=6006</guid>
		<description><![CDATA[本文永久链接 &#8211; https://tonybai.com/2026/03/08/her-power-in-code-pioneers-to-ai-era 大家好，我是Tony Bai。 当我们闭上眼睛，想象一个“程序员”的形象时，脑海中浮现的画面是什么？ 很长一段时间里，流行文化和媒体在不遗余力地塑造一种刻板印象：穿着格子衬衫、戴着黑框眼镜、不善言辞的男性，在昏暗的灯光下敲击着键盘。硅谷的“兄弟会文化（Bro-culture）”更是将这种刻板印象固化，仿佛编程从诞生之日起，就是一项由男性绝对主导的活动。 然而，如果我们翻开计算机科学的真实历史，会发现一个令人惊讶，甚至有些反直觉的事实：在计算机刚刚诞生的黎明期，编程，曾经是一项被普遍认为“适合女性”的工作。 在二战期间，由于男性大量奔赴前线，世界上第一台通用电子计算机 ENIAC 的初代程序员团队，清一色全是由六位杰出的女性组成。她们在没有编程语言、没有编译器的时代，用插拔线缆和拨动开关的纯物理方式，完成了极其复杂的弹道轨迹计算。 然而，随着软件产业的爆炸式增长，薪资与地位水涨船高，女性在科技行业的比例却开始出现诡异的下滑，她们的名字也逐渐被隐藏在庞大服务器的阴影之中。 今天是 3 月 8 日国际妇女节。在这个特殊的日子里，让我们暂时停下手中正在 Review 的代码，去擦拭掉历史上的偏见灰尘。我们要重新认识那些在计算机科学发展史上立下不朽丰碑的女性先驱，看看当今站在技术浪潮之巅的领航者，并探讨在汹涌而来的 AI 时代，“巾帼力量”为何比以往任何时候都更加不可或缺。 历史丰碑：她们写下了改变世界的最初几行代码 代码是没有性别的，但在计算机还是一堆庞大齿轮或真空管的年代，是这些女性赋予了冷冰冰的机器以“逻辑的灵魂”。 “诗意科学”的先知：Ada Lovelace（埃达·洛夫莱斯） 要追溯程序员的祖师爷，我们必须回到 19 世纪中叶的英国。著名诗人拜伦的女儿，Ada Lovelace，被公认为世界上的第一位程序员。 当时的数学家查尔斯·巴贝奇正在设计一台名为“分析机”的庞大机械装置。在多数人看来，这只是一个能做加减乘除的超大号计算器。但 Ada 展现出了超越时代一个世纪的惊人洞察力。 在翻译和注释关于分析机的文章时，她不仅写下了世界上第一段计算机算法（用于计算伯努利数），更重要的是，她写下了一段堪称“预言”的批注。Ada 指出，如果分析机能够处理数字，那么只要将事物（如字母、音乐）转化为数字，机器就能处理任何事物。 “分析机编织的是代数模式，就像提花织机编织树叶和花朵一样。” 这是一种被称为“诗意科学”的浪漫与理性的结合。Ada 早在计算机诞生前 100 年，就看透了现代计算机的本质：它不仅仅是计算工具，而是通用的信息处理引擎。今天美国国防部开发的 Ada 语言，正是为了纪念这位伟大的女性“先知”。 编译器的鼻祖与“捉虫”专家：Grace Hopper（格蕾丝·霍珀） 如果说 Ada 给出了灵魂，那么 Grace Hopper 则是真正让机器“听懂”人类语言的架构师。 在 20 世纪 50 年代，程序员们必须用极其难懂的二进制机器码来编写指令。这种方式不仅痛苦，而且极易出错。Hopper 坚信，程序员应该能够用接近英语的语言来编写代码，然后再由机器自己将其翻译成机器码。 [...]]]></description>
			<content:encoded><![CDATA[<p><img src="https://tonybai.com/wp-content/uploads/2026/her-power-in-code-pioneers-to-ai-era-1.png" alt="" /></p>
<p><a href="https://tonybai.com/2026/03/08/her-power-in-code-pioneers-to-ai-era">本文永久链接</a> &#8211; https://tonybai.com/2026/03/08/her-power-in-code-pioneers-to-ai-era</p>
<p>大家好，我是Tony Bai。</p>
<p>当我们闭上眼睛，想象一个“程序员”的形象时，脑海中浮现的画面是什么？</p>
<p>很长一段时间里，流行文化和媒体在不遗余力地塑造一种刻板印象：穿着格子衬衫、戴着黑框眼镜、不善言辞的男性，在昏暗的灯光下敲击着键盘。硅谷的“兄弟会文化（Bro-culture）”更是将这种刻板印象固化，仿佛编程从诞生之日起，就是一项由男性绝对主导的活动。</p>
<p>然而，如果我们翻开计算机科学的真实历史，会发现一个令人惊讶，甚至有些反直觉的事实：<strong>在计算机刚刚诞生的黎明期，编程，曾经是一项被普遍认为“适合女性”的工作。</strong></p>
<p>在二战期间，由于男性大量奔赴前线，世界上第一台通用电子计算机 ENIAC 的初代程序员团队，清一色全是由六位杰出的女性组成。她们在没有编程语言、没有编译器的时代，用插拔线缆和拨动开关的纯物理方式，完成了极其复杂的弹道轨迹计算。</p>
<p>然而，随着软件产业的爆炸式增长，薪资与地位水涨船高，女性在科技行业的比例却开始出现诡异的下滑，她们的名字也逐渐被隐藏在庞大服务器的阴影之中。</p>
<p>今天是 3 月 8 日国际妇女节。在这个特殊的日子里，让我们暂时停下手中正在 Review 的代码，去擦拭掉历史上的偏见灰尘。我们要重新认识那些在计算机科学发展史上立下不朽丰碑的女性先驱，看看当今站在技术浪潮之巅的领航者，并探讨在汹涌而来的 AI 时代，“巾帼力量”为何比以往任何时候都更加不可或缺。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2026/agentic-software-engineering-qr.png" alt="" /></p>
<h2>历史丰碑：她们写下了改变世界的最初几行代码</h2>
<p>代码是没有性别的，但在计算机还是一堆庞大齿轮或真空管的年代，是这些女性赋予了冷冰冰的机器以“逻辑的灵魂”。</p>
<h3>“诗意科学”的先知：Ada Lovelace（埃达·洛夫莱斯）</h3>
<p><img src="https://tonybai.com/wp-content/uploads/2026/her-power-in-code-pioneers-to-ai-era-2.png" alt="" /></p>
<p>要追溯程序员的祖师爷，我们必须回到 19 世纪中叶的英国。著名诗人拜伦的女儿，Ada Lovelace，被公认为世界上的<strong>第一位程序员</strong>。</p>
<p>当时的数学家查尔斯·巴贝奇正在设计一台名为“分析机”的庞大机械装置。在多数人看来，这只是一个能做加减乘除的超大号计算器。但 Ada 展现出了超越时代一个世纪的惊人洞察力。</p>
<p>在翻译和注释关于分析机的文章时，她不仅写下了世界上第一段计算机算法（用于计算伯努利数），更重要的是，她写下了一段堪称“预言”的批注。Ada 指出，如果分析机能够处理数字，那么只要将事物（如字母、音乐）转化为数字，机器就能处理任何事物。</p>
<blockquote>
<p>“分析机编织的是代数模式，就像提花织机编织树叶和花朵一样。”</p>
</blockquote>
<p>这是一种被称为“诗意科学”的浪漫与理性的结合。Ada 早在计算机诞生前 100 年，就看透了现代计算机的本质：它不仅仅是计算工具，而是通用的信息处理引擎。今天美国国防部开发的 Ada 语言，正是为了纪念这位伟大的女性“先知”。</p>
<h3>编译器的鼻祖与“捉虫”专家：Grace Hopper（格蕾丝·霍珀）</h3>
<p><img src="https://tonybai.com/wp-content/uploads/2026/her-power-in-code-pioneers-to-ai-era-3.png" alt="" /></p>
<p>如果说 Ada 给出了灵魂，那么 Grace Hopper 则是真正让机器“听懂”人类语言的架构师。</p>
<p>在 20 世纪 50 年代，程序员们必须用极其难懂的二进制机器码来编写指令。这种方式不仅痛苦，而且极易出错。Hopper 坚信，程序员应该能够用接近英语的语言来编写代码，然后再由机器自己将其翻译成机器码。</p>
<p>当她提出这个想法时，遭到了几乎所有同行的嘲笑和拒绝。他们认为“计算机只能懂数字，不可能懂英语”。但 Hopper 是一位拥有美国海军准将军衔的“硬核”女性，她顶住了所有压力，成功开发出了世界上<strong>第一个编译器 A-0</strong>，并直接主导了后来统治商业系统数十年的 COBOL 语言的诞生。</p>
<p>除了这项伟大的技术发明，Hopper 还给全世界程序员留下了一个最常用的口头禅。1947 年，她在哈佛大学的一台继电器计算机里发现了一只导致故障的真实飞蛾（Moth）。她将这只飞蛾粘在日志本上，并在旁边写下：“First actual case of bug being found.（发现的第一个真正的 Bug）”。从此，程序员排查错误的过程，就永远被称为了 “Debug”（除虫）。</p>
<h3>登月背后的无名英雄：Margaret Hamilton（玛格丽特·汉密尔顿）</h3>
<p><img src="https://tonybai.com/wp-content/uploads/2026/her-power-in-code-pioneers-to-ai-era-4.png" alt="" /></p>
<p>有一张在科技史流传甚广的照片：一位年轻的戴着大框眼镜的女性，微笑着站在一堆比她自己还要高的打印源代码旁。她就是 Margaret Hamilton，阿波罗 11 号登月计划的首席软件工程师。</p>
<p>在 1969 年那个登月舱只有几十 KB 内存的年代，写代码绝不容许有任何试错的空间。更重要的是，在那个年代，“软件”甚至不被认为是一门严谨的工程学科。是 Hamilton 第一次创造了 <strong>“软件工程 (Software Engineering)”</strong> 这个词，并为其赋予了与硬件工程同等的严谨性。</p>
<p>她的远见卓识在历史性的一刻拯救了全人类的心跳。就在阿波罗 11 号即将降落月球表面的最后 3 分钟，由于雷达系统的硬件故障，登月舱的计算机突然被大量无关的数据淹没，系统濒临崩溃，警报声大作。</p>
<p>在地面指挥中心准备下令中止登月时，Hamilton 带领团队设计的<strong>“异步优先调度（Asynchronous Executive）”机制</strong>发挥了奇效。这段极其健壮的容错代码，让计算机瞬间抛弃了低优先级的雷达任务，将全部仅存的算力集中在最关键的着陆控制上。</p>
<p>阿姆斯特朗成功踏上了月球，而这背后，是 Hamilton 用代码织就的绝对安全网。</p>
<h2>当代灯塔：站在技术浪潮之巅的开源与企业领袖</h2>
<p>历史的丰碑固然闪耀，但“巾帼力量”绝不仅仅存在于泛黄的黑白照片中。当我们把视线拉回当代，你会发现在云计算、开源社区和最前沿的人工智能领域，女性依然是不可或缺的领航者。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2026/her-power-in-code-pioneers-to-ai-era-5.png" alt="" /></p>
<p>在开源世界的深水区，也就是最具“硬核极客文化”的容器和底层基础设施领域，Jessie Frazelle 的名字如雷贯耳。作为 Docker 的核心维护者之一，她写下了 Docker 中许多最底层的安全和隔离特性代码。她以一人之力在充满偏见和偶尔充斥着“有毒（Toxic）”言论的开源社区中杀出一条血路，证明了女性同样可以在最底层的系统编程中达到登峰造极的水平。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2026/her-power-in-code-pioneers-to-ai-era-6.png" alt="" /></p>
<p>而在当今如火如荼的 AI 浪潮中，我们更不能忘记李飞飞 (Fei-Fei Li)。在深度学习还处于被学术界边缘化的低谷期时，李飞飞敏锐地意识到：模型再好，没有海量的高质量数据也无法发生质变。于是，她顶住巨大压力，发起了 ImageNet 计划，构建了一个包含 1400 万张标注图片的庞大数据库。</p>
<p>正是 ImageNet 的存在，直接催生了 2012 年 AlexNet 的横空出世，引发了这一轮浩浩荡荡的深度学习和 AI 大爆发。她被称为“AI 界的拓荒者”，用女性特有的坚韧和长远目光，为整个行业打下了最坚实的地基。</p>
<h2>AI 时代的新契机：为什么未来的技术世界更需要“她”？</h2>
<p>2024 年至今，随着生成式 AI（GenAI）、大型语言模型（LLM）以及自主 Agent（如 Claude Code, Cursor）的极速普及，<a href="https://tonybai.com/2026/02/28/agentic-software-engineering">软件工程的范式</a>正在经历一场彻底的颠覆。</p>
<p>“敲击代码”这一纯体力的动作正在被 AI 代替。很多从业者感到恐慌：如果机器能在几秒钟内写出完美的并发处理代码，程序员的价值到底在哪里？</p>
<p>讽刺的是，这场由机器主导的技术革命，反而为女性程序员在科技行业中的地位跃升，提供了百年难遇的新契机。为什么这么说？</p>
<h3>从“机器语者”到“交响乐指挥”：Prompt 工程与沟通的艺术</h3>
<p>在传统的编程时代，程序员需要像机器一样思考，用极其死板和严苛的语法去迎合编译器。这在某种程度上，筛选出了一批极度专注于逻辑细节、但不一定擅长横向沟通的人群。</p>
<p>但在 AI 辅助编程时代，人类的角色从“写代码的工人”变成了“指挥 AI 的产品经理”。你需要做的是深刻理解业务需求、拆解复杂系统，并用自然语言（Prompt）精准地将意图传达给 AI。</p>
<p>这要求极高的<strong>沟通能力、同理心、大局观以及对模糊意图的澄清能力</strong>。而这些，恰恰是许多女性在长期社会化过程中被培养出的显著优势。未来的顶级工程师，不再是那些能背诵冷门 API 的人，而是那些能够清晰表达意图、优雅编排多个 AI Agent 协同工作的“交响乐指挥”。</p>
<h3>消除算法的“傲慢与偏见”：AI 伦理的守门人</h3>
<p>AI 就像一面镜子，它会无情地反射并放大人类社会中存在的所有偏见。如果我们训练 AI 模型的工程师团队是清一色的单一性别、单一族裔（例如传统的“硅谷白人男性俱乐部”），那么这个 AI 生成的简历筛选算法、医疗诊断模型或是自动驾驶策略，必然会带有难以察觉的系统性偏见。</p>
<p>在 AI 对齐（Alignment）和 AI 安全（AI Safety）领域，我们需要多元化的视角来纠正机器的偏见。女性研究者和工程师在感知社会公平、识别弱势群体需求方面往往具有更敏锐的触觉。如今，在 OpenAI、Anthropic 等顶级 AI 实验室中，主导 AI 伦理和安全护栏工作的核心领导层中，出现了越来越多卓越的女性身影。比如Anthropic联合创始人阿曼达·阿斯克尔（Amanda Askell），就是一位训练有素的哲学家，她帮助管理Claude的个性。<strong>没有女性参与的 AI，注定是一个有缺陷的 AI。</strong></p>
<h3>全栈通才的崛起与“产品思维”的胜利</h3>
<p>由于 AI 极大地降低了后端的复杂度和前端页面的构建门槛，“一人公司”或“超级小团队”正在成为现实。</p>
<p>这要求未来的开发者必须是懂产品、懂设计、懂用户心理的“全栈通才”。仅仅会写高并发代码已经不够了，你还需要知道如何设计出让用户感到温暖、舒适的交互界面。女性往往具备更强的跨界融合能力和细腻的用户感知能力，在“技术与人文的十字路口”，她们将比纯粹的“代码机器”爆发出更强大的创造力。</p>
<h2>小结：传承遗产，编写未来</h2>
<p>回顾历史，从 Ada Lovelace 描绘在纸带上的第一个循环，到 Grace Hopper 拔出的第一只真实飞蛾；从 Margaret Hamilton 保护阿波罗登月的汇编指令，到如今女性工程师在 LLM 底层写的对齐代码。</p>
<p><strong>女性，从未在计算机科学的历史中缺席。</strong> 她们不仅是历史的参与者，更是很多决定性瞬间的缔造者。</p>
<p>然而，我们依然要清醒地看到，今天在 GitHub 的开源提交中、在科技公司的高管会议室里，女性的比例依然没有达到应有的平衡。打破这种隐形的“天花板”和玻璃墙，需要我们每一个人——无论男女——去对抗潜意识中的刻板印象。</p>
<p>代码没有性别，Bug 也不分男女。优秀的架构设计只认同逻辑的严密，而不关心键盘后那双手的粗细。</p>
<p>在这个 AI 浪潮奔涌的时代前夕，让我们向所有奋斗在键盘前、熬夜在服务器旁、在开源社区里无私贡献的女程序员们致以最崇高的敬意。</p>
<p>愿 Ada 的远见、Hopper 的坚持和 Hamilton 的严谨，能够化作一行行永不退色的代码，注入到每一位女性开发者的指尖。</p>
<p><strong>3.8 国际妇女节快乐！愿你们继续用代码，勇敢、自由地编译属于你们的未来！</strong></p>
<hr />
<p><strong>致敬身边的“她”</strong></p>
<p>在你的开发生涯中，是否曾遇到过让你深感佩服的女性技术伙伴？或者，作为一名女性开发者，你在 AI 时代的浪潮中有什么独特的感悟？</p>
<p>欢迎在评论区留下你对“她”的赞美或故事！我们将精选留言，一起传递这份力量。</p>
<hr />
<p>还在为“复制粘贴喂AI”而烦恼？我的新专栏 <strong>《<a href="http://gk.link/a/12EPd">AI原生开发工作流实战</a>》</strong> 将带你：</p>
<ul>
<li>告别低效，重塑开发范式</li>
<li>驾驭AI Agent(Claude Code)，实现工作流自动化</li>
<li>从“AI使用者”进化为规范驱动开发的“工作流指挥家”</li>
</ul>
<p>扫描下方二维码，开启你的AI原生开发之旅。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/ai-native-dev-workflow-qr.png" alt="" /></p>
<hr />
<p>你的Go技能，是否也卡在了“熟练”到“精通”的瓶颈期？</p>
<ul>
<li>想写出更地道、更健壮的Go代码，却总在细节上踩坑？</li>
<li>渴望提升软件设计能力，驾驭复杂Go项目却缺乏章法？</li>
<li>想打造生产级的Go服务，却在工程化实践中屡屡受挫？</li>
</ul>
<p>继《<a href="http://gk.link/a/10AVZ">Go语言第一课</a>》后，我的《<a href="http://gk.link/a/12yGY">Go语言进阶课</a>》终于在极客时间与大家见面了！</p>
<p>我的全新极客时间专栏 《<a href="http://gk.link/a/12yGY">Tony Bai·Go语言进阶课</a>》就是为这样的你量身打造！30+讲硬核内容，带你夯实语法认知，提升设计思维，锻造工程实践能力，更有实战项目串讲。</p>
<p>目标只有一个：助你完成从“Go熟练工”到“Go专家”的蜕变！ 现在就加入，让你的Go技能再上一个新台阶！</p>
<p><img src="https://tonybai.com/wp-content/uploads/course-card/iamtonybai-banner-2.gif" alt="" /></p>
<hr />
<p><strong>原「Gopher部落」已重装升级为「Go &amp; AI 精进营」知识星球，快来加入星球，开启你的技术跃迁之旅吧！</strong></p>
<p>我们致力于打造一个高品质的 <strong>Go 语言深度学习</strong> 与 <strong>AI 应用探索</strong> 平台。在这里，你将获得：</p>
<ul>
<li><strong>体系化 Go 核心进阶内容:</strong> 深入「Go原理课」、「Go进阶课」、「Go避坑课」等独家深度专栏，夯实你的 Go 内功。</li>
<li><strong>前沿 Go+AI 实战赋能:</strong> 紧跟时代步伐，学习「Go+AI应用实战」、「Agent开发实战课」、「Agentic软件工程课」、「Claude Code开发工作流实战课」、「OpenClaw实战分享」等，掌握 AI 时代新技能。 </li>
<li><strong>星主 Tony Bai 亲自答疑:</strong> 遇到难题？星主第一时间为你深度解析，扫清学习障碍。</li>
<li><strong>高活跃 Gopher 交流圈:</strong> 与众多优秀 Gopher 分享心得、讨论技术，碰撞思想火花。</li>
<li><strong>独家资源与内容首发:</strong> 技术文章、课程更新、精选资源，第一时间触达。</li>
</ul>
<p>衷心希望「Go &amp; AI 精进营」能成为你学习、进步、交流的港湾。让我们在此相聚，享受技术精进的快乐！欢迎你的加入！</p>
<p><img src="http://image.tonybai.com/img/tonybai/gopher-and-ai-tribe-zsxq-small-card.jpg" alt="img{512x368}" /></p>
<hr />
<p><strong>想系统学习Go，构建扎实的知识体系？</strong></p>
<p>我的新书《<a href="https://book.douban.com/subject/37499496/">Go语言第一课</a>》是你的首选。源自2.4万人好评的极客时间专栏，内容全面升级，同步至Go 1.24。首发期有专属五折优惠，不到40元即可入手，扫码即可拥有这本300页的Go语言入门宝典，即刻开启你的Go语言高效学习之旅！</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/go-primer-published-4.png" alt="" /></p>
<hr />
<p>商务合作方式：撰稿、出书、培训、在线课程、合伙创业、咨询、广告合作。如有需求，请扫描下方公众号二维码，与我私信联系。</p>
<p><img src="http://image.tonybai.com/img/tonybai/iamtonybai-wechat-qr.png" alt="" /></p>
<p style='text-align:left'>&copy; 2026, <a href='https://tonybai.com'>bigwhite</a>. 版权所有. </p>
]]></content:encoded>
			<wfw:commentRss>https://tonybai.com/2026/03/08/her-power-in-code-pioneers-to-ai-era/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>TypeScript 编译器 Go 重写版提速 10 倍：微软团队深度揭秘幕后工程细节</title>
		<link>https://tonybai.com/2026/01/27/typescript-compiler-go-rewrite-10x-speed-microsoft-details/</link>
		<comments>https://tonybai.com/2026/01/27/typescript-compiler-go-rewrite-10x-speed-microsoft-details/#comments</comments>
		<pubDate>Mon, 26 Jan 2026 23:21:49 +0000</pubDate>
		<dc:creator>bigwhite</dc:creator>
				<category><![CDATA[技术志]]></category>
		<category><![CDATA[AbstractSyntaxTree]]></category>
		<category><![CDATA[ast]]></category>
		<category><![CDATA[async]]></category>
		<category><![CDATA[Compiler]]></category>
		<category><![CDATA[Concurrency]]></category>
		<category><![CDATA[Consstring]]></category>
		<category><![CDATA[CrossPlatform]]></category>
		<category><![CDATA[electron]]></category>
		<category><![CDATA[Figma]]></category>
		<category><![CDATA[FunctionColoring]]></category>
		<category><![CDATA[GC]]></category>
		<category><![CDATA[GenericMethods]]></category>
		<category><![CDATA[Go]]></category>
		<category><![CDATA[GopherCon2025]]></category>
		<category><![CDATA[IndependentCheckers]]></category>
		<category><![CDATA[JakeBailey]]></category>
		<category><![CDATA[JIT优化]]></category>
		<category><![CDATA[MemoryManagement]]></category>
		<category><![CDATA[Microsoft]]></category>
		<category><![CDATA[NativeCode]]></category>
		<category><![CDATA[NilSafety]]></category>
		<category><![CDATA[node.js]]></category>
		<category><![CDATA[NullCoalescing]]></category>
		<category><![CDATA[OOM]]></category>
		<category><![CDATA[OptionalChaining]]></category>
		<category><![CDATA[PerformanceLeap]]></category>
		<category><![CDATA[Rewrite]]></category>
		<category><![CDATA[Selfhosting]]></category>
		<category><![CDATA[shadowing]]></category>
		<category><![CDATA[SharedMemory]]></category>
		<category><![CDATA[slack]]></category>
		<category><![CDATA[StructEmbedding]]></category>
		<category><![CDATA[TernaryOperator]]></category>
		<category><![CDATA[ts-to-go]]></category>
		<category><![CDATA[TypeChecking]]></category>
		<category><![CDATA[TypeScript]]></category>
		<category><![CDATA[TypeScript7.0]]></category>
		<category><![CDATA[UnionTypes]]></category>
		<category><![CDATA[vscode]]></category>
		<category><![CDATA[wasm]]></category>
		<category><![CDATA[WebAssembly]]></category>
		<category><![CDATA[三元运算符]]></category>
		<category><![CDATA[上位]]></category>
		<category><![CDATA[优化]]></category>
		<category><![CDATA[共享内存]]></category>
		<category><![CDATA[内存拷贝]]></category>
		<category><![CDATA[内存管理]]></category>
		<category><![CDATA[内存限制]]></category>
		<category><![CDATA[函数着色]]></category>
		<category><![CDATA[单线程]]></category>
		<category><![CDATA[垃圾回收]]></category>
		<category><![CDATA[多核CPU]]></category>
		<category><![CDATA[字符串拼接]]></category>
		<category><![CDATA[学习曲线]]></category>
		<category><![CDATA[实用主义]]></category>
		<category><![CDATA[对象模型]]></category>
		<category><![CDATA[工程挑战]]></category>
		<category><![CDATA[平滑上手]]></category>
		<category><![CDATA[并发设计]]></category>
		<category><![CDATA[开发成本]]></category>
		<category><![CDATA[异步]]></category>
		<category><![CDATA[影子变量]]></category>
		<category><![CDATA[微软]]></category>
		<category><![CDATA[心智负担]]></category>
		<category><![CDATA[性能天花板]]></category>
		<category><![CDATA[性能提升]]></category>
		<category><![CDATA[技术细节]]></category>
		<category><![CDATA[数据结构]]></category>
		<category><![CDATA[机器码]]></category>
		<category><![CDATA[泛型方法]]></category>
		<category><![CDATA[测试用例]]></category>
		<category><![CDATA[独立检查器]]></category>
		<category><![CDATA[电子表格]]></category>
		<category><![CDATA[破坏性变更]]></category>
		<category><![CDATA[空值合并]]></category>
		<category><![CDATA[空值安全]]></category>
		<category><![CDATA[符号绑定]]></category>
		<category><![CDATA[等价交换]]></category>
		<category><![CDATA[类型检查]]></category>
		<category><![CDATA[结构体嵌入]]></category>
		<category><![CDATA[编译器]]></category>
		<category><![CDATA[联合类型]]></category>
		<category><![CDATA[胖指针]]></category>
		<category><![CDATA[自举]]></category>
		<category><![CDATA[自动化工具]]></category>
		<category><![CDATA[范本]]></category>
		<category><![CDATA[设计]]></category>
		<category><![CDATA[语法糖]]></category>
		<category><![CDATA[跨平台]]></category>
		<category><![CDATA[软件规范]]></category>
		<category><![CDATA[运行速度]]></category>
		<category><![CDATA[重写]]></category>
		<category><![CDATA[链式调用]]></category>
		<category><![CDATA[静态分析]]></category>

		<guid isPermaLink="false">https://tonybai.com/?p=5776</guid>
		<description><![CDATA[本文永久链接 &#8211; https://tonybai.com/2026/01/27/typescript-compiler-go-rewrite-10x-speed-microsoft-details 大家好，我是Tony Bai。 “JavaScript 是一门很棒的语言，但它并不是为了编写编译器而设计的。” 备受瞩目的 TypeScript 编译器 Go 重写版（代号 TypeScript 7.0）已经取得了惊人的 10 倍性能提升。在最近的 GopherCon 2025 上，来自 Microsoft TypeScript 团队的 Jake Bailey 带来了一场干货满满的分享，深度揭秘了这场跨语言大迁徙背后的工程挑战与技术细节。 为什么最终选择了 Go？庞大的 AST 如何在 Go 中高效表达？又是如何通过并发设计打破 Node.js 的性能枷锁的？本文将带你深入编译器内部，一探究竟。 缘起：当 JavaScript 触碰到天花板 TypeScript 自 2012 年发布以来，一直采用“自举” (Self-hosting) 的方式，即用 TypeScript 编写 TypeScript 编译器。这带来了巨大的好处：团队能第一时间吃自己的狗粮，社区贡献也极其方便。 然而，JavaScript 并不是为了编写高性能编译器而设计的。随着代码库规模的爆炸式增长（如 VS Code 的 150 万行代码），基于 Node.js 的编译器逐渐触碰到了性能天花板： [...]]]></description>
			<content:encoded><![CDATA[<p><img src="https://tonybai.com/wp-content/uploads/2026/typescript-compiler-go-rewrite-10x-speed-microsoft-details-1.png" alt="" /></p>
<p><a href="https://tonybai.com/2026/01/27/typescript-compiler-go-rewrite-10x-speed-microsoft-details">本文永久链接</a> &#8211; https://tonybai.com/2026/01/27/typescript-compiler-go-rewrite-10x-speed-microsoft-details</p>
<p>大家好，我是Tony Bai。</p>
<p>“JavaScript 是一门很棒的语言，但它并不是为了编写编译器而设计的。”</p>
<p>备受瞩目的 <a href="https://tonybai.com/2025/03/13/interview-with-anders-hejlsberg">TypeScript 编译器 Go 重写版</a>（代号 TypeScript 7.0）已经取得了惊人的 10 倍性能提升。在最近的 GopherCon 2025 上，来自 Microsoft TypeScript 团队的 Jake Bailey 带来了<a href="https://www.youtube.com/watch?v=PZm_YbE3fcA">一场干货满满的分享</a>，深度揭秘了这场跨语言大迁徙背后的工程挑战与技术细节。</p>
<p>为什么最终选择了 Go？庞大的 AST 如何在 Go 中高效表达？又是如何通过并发设计打破 Node.js 的性能枷锁的？本文将带你深入编译器内部，一探究竟。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2026/distributed-system-guide-qr.png" alt="img{512x368}" /></p>
<h2>缘起：当 JavaScript 触碰到天花板</h2>
<p>TypeScript 自 2012 年发布以来，一直采用“自举” (Self-hosting) 的方式，即用 TypeScript 编写 TypeScript 编译器。这带来了巨大的好处：团队能第一时间吃自己的狗粮，社区贡献也极其方便。</p>
<p>然而，JavaScript 并不是为了编写高性能编译器而设计的。随着代码库规模的爆炸式增长（如 VS Code 的 150 万行代码），基于 Node.js 的编译器逐渐触碰到了性能天花板：</p>
<ul>
<li><strong>单线程与内存限制</strong>：JavaScript 无法高效利用多核 CPU，且 Node.js 构建环境（如 Electron）常常面临 4GB 内存上限，导致大型项目编译时频繁 OOM。</li>
<li><strong>昂贵的对象模型</strong>：JavaScript 的对象模型开销巨大，而编译器需要创建数以百万计的 AST 节点，这对内存和 GC 都是沉重的负担。</li>
<li><strong>异步的代价</strong>：async/await 虽然方便，但带来了著名的“函数着色”问题，且 Promise 对象的分配本身就有非零的运行时开销。</li>
</ul>
<p><img src="https://tonybai.com/wp-content/uploads/2026/typescript-compiler-go-rewrite-10x-speed-microsoft-details-2.png" alt="" /></p>
<p>尽管团队已经用尽了 JIT 优化、缓存、单态化 (monomorphization) 等高级手段，但性能提升的边际效应越来越小，OOM 问题依然挥之不去。移植到另外一种语言，成为了打破僵局的唯一选择。</p>
<h2>明确目标：新编译器的硬性指标</h2>
<p>既然决定要移植到新语言，那么新语言必须解决 JavaScript 的痛点，同时不能丢失现有的优势。团队列出了几条不可妥协的硬性指标：</p>
<ol>
<li><strong>极致速度</strong>：必须编译为原生机器码 (Native Code)，摆脱解释器和 JIT 的预热开销。</li>
<li><strong>共享内存并发</strong>：这是性能翻盘的关键。新语言必须对多线程共享内存有强力支持，以便充分压榨多核性能。</li>
<li><strong>跨平台支持</strong>：必须能运行在所有主流操作系统上，最重要的是——<strong>必须能编译为 WebAssembly</strong>，以确保在浏览器环境（如 vscode.dev）中的体验。</li>
<li><strong>无缝移植</strong>：鉴于 TypeScript 没有正式的语言规范（Spec），现有的编译器实现就是事实上的规范。因此，新语言必须能够最大程度地保留原有代码的结构和逻辑，以确保行为的一致性。</li>
</ol>
<p>正是这几条苛刻的标准，将选型的范围迅速缩小。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2026/typescript-compiler-go-rewrite-10x-speed-microsoft-details-4.png" alt="" /></p>
<h2>选型：为什么是 Go？</h2>
<p><img src="https://tonybai.com/wp-content/uploads/2026/typescript-compiler-go-rewrite-10x-speed-microsoft-details-3.png" alt="" /></p>
<p>在考察了 Rust、C#、Zig 等语言后，Go 脱颖而出。Jake 透露了核心的决策逻辑：</p>
<ol>
<li><strong>带 GC 的内存管理</strong>：编译器涉及大量复杂的、循环引用的数据结构（如 AST 节点），“手动”管理内存（如 Rust）会带来巨大的心智负担和开发成本。Go 的 GC 完美契合这一需求。</li>
<li><strong>结构相似性</strong>：TypeScript 的代码风格（无类、大量函数和接口）与 Go 非常相似。这使得“移植”而非“重写”成为可能。</li>
<li><strong>学习曲线平缓</strong>：团队中大部分是 TypeScript 专家而非系统编程专家。Go 的简单性让团队能迅速上手。</li>
<li><strong>跨平台与性能</strong>：Go 编译为原生机器码，天生支持高并发，且能轻松跨平台（包括编译为 WASM）。</li>
</ol>
<p>Go完美地契合了TypeScript编译器移植的需求！</p>
<p><img src="https://tonybai.com/wp-content/uploads/2026/typescript-compiler-go-rewrite-10x-speed-microsoft-details-5.png" alt="" /></p>
<h2>早期验证：手写原型与意外惊喜</h2>
<p>在决定全面转向 Go 之前，团队并未贸然行动，而是采取了稳健的“原型验证”策略。</p>
<p>他们从编译器的最底层——<strong>扫描器 (Scanner) 和解析器 (Parser)</strong>——开始，尝试手工将 TypeScript 代码逐行“翻译”为 Go 代码。与此同时，为了确保决策万无一失，还有几位成员试探性地尝试了其他语言方案。</p>
<p>结果令人振奋：即使是初步的手写 Go 代码，<strong>解析速度也达到了原版的 5 倍左右！</strong></p>
<p>更重要的是，团队惊喜地发现，<strong>手写的 Go 代码在结构和逻辑上与原始的 TypeScript 代码惊人地相似</strong>。这种代码形态上的高度一致性，不仅验证了 Go 是正确的选择，更为后续大规模自动化工具的开发注入了强心剂。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2026/typescript-compiler-go-rewrite-10x-speed-microsoft-details-7.png" alt="" /></p>
<h2>移植实战：从 ts-to-go 到并发革命</h2>
<h3>1. 自动化移植工具：ts-to-go</h3>
<p>为了加速迁移，Jake 编写了一个 <a href="https://github.com/jakebailey/ts-to-go">ts-to-go</a> 工具，能将 TypeScript 代码“直译”为 Go 代码。</p>
<ul>
<li>TS 的 interface -> Go 的 interface</li>
<li>TS 的 class -> Go 的 struct + methods</li>
<li>复杂的位运算和逻辑判断 -> 自动转换为 Go 的等价写法</li>
</ul>
<p>虽然不能 100% 完美转换，但这让团队在初期就能获得一个“虽然丑但能跑”的版本，极大加速了进程。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2026/typescript-compiler-go-rewrite-10x-speed-microsoft-details-6.png" alt="" /></p>
<h3>2. 数据结构的重新设计</h3>
<p>在 JavaScript 中，对象是动态的；在 Go 中，一切皆有类型。团队不得不对 AST 的数据结构进行大刀阔斧的改革。</p>
<ul>
<li><strong>消除 interface 滥用</strong>：最初的移植版本大量使用 interface 来模拟 TS 的多态，导致了巨大的内存开销（胖指针）和 nil 检查地狱。</li>
<li><strong>拥抱 struct 嵌入</strong>：最终，他们设计了一个基础 Node 结构体，并将其嵌入到所有具体的 AST 节点中。这不仅减少了内存占用，还彻底解决了 nil 接口的问题。</li>
</ul>
<h3>3. 并发：性能提升的核心引擎</h3>
<p>这是 Go 带来的最大红利。旧的 TS 编译器是单线程的，解析、绑定、检查、生成都在一条线上排队。</p>
<p>而在 Go 版本中：</p>
<ul>
<li><strong>解析 (Parsing)</strong>：每个文件可以独立解析，完全并行。</li>
<li><strong>绑定 (Binding)</strong>：每个文件的符号绑定也是独立的，完全并行。</li>
<li><strong>类型检查 (Type Checking)</strong>：这是最难的部分，因为文件间存在复杂的依赖。团队采用了<strong>“独立检查器” (Independent Checkers)</strong> 的模式，为每组文件分配一个独立的检查器，虽然会有少量重复工作，但实现了高度的并行化。</li>
</ul>
<p><img src="https://tonybai.com/wp-content/uploads/2026/typescript-compiler-go-rewrite-10x-speed-microsoft-details-9.png" alt="" /></p>
<p>结果是惊人的：<strong>VS Code 的编译时间从 80 秒缩短到了 7 秒，速度提升超过 10 倍！</strong></p>
<p><img src="https://tonybai.com/wp-content/uploads/2026/typescript-compiler-go-rewrite-10x-speed-microsoft-details-8.png" alt="" /></p>
<h2>踩坑与优化：Go 也没那么简单</h2>
<p>当然，移植过程并非一帆风顺。Jake 分享了几个典型的“水土不服”案例：</p>
<ul>
<li><strong>影子变量 (Shadowing)</strong>：Go 允许在内层作用域遮蔽外层变量（如 err、result等），这导致了无数隐蔽的 Bug。Jake 甚至为此专门写了一个静态分析工具(https://jakebailey.dev/posts/go-shadowing)来抓这些虫子。</li>
<li><strong>方法值的分配</strong>：在 Go 中，将方法作为值传递（如 parser.LookAhead）会产生一次内存分配。在一个频繁调用的紧密循环中，这带来了 17% 的性能损耗。解决方案是改回显式的函数调用。</li>
<li><strong>字符串拼接</strong>：JavaScript 引擎对字符串拼接有深度优化（Cons-string），而 Go 的 + 操作符则是实打实的内存拷贝。这导致初期的移植版本在处理大量字符串时性能惨不忍睹。</li>
</ul>
<h2>遗憾与取舍：那些我们怀念的 TypeScript 特性</h2>
<p>正如 Jake 在演讲中所言，这次迁移是一场巨大的工程胜利，但也是一次充满妥协的旅程。从表达力丰富的 TypeScript 转向“极简主义”的 Go，团队不得不忍痛割爱，放弃了许多令人怀念的语言特性：</p>
<ul>
<li><strong>编译期空值安全 (Compile-time nil safety)</strong>：这是团队最怀念的特性。在 Go 中，空指针异常（Panic）依然是悬在头顶的达摩克利斯之剑，而在 TypeScript 中，null/undefined 是类型系统的一部分，能被编译器严格检查。</li>
<li><strong>空值合并与链式调用 (??, ?.)</strong>：Go 缺乏这些语法糖，使得代码中充斥着冗长的 if x != nil 检查，远不如 TypeScript 优雅。</li>
<li><strong>联合类型与类型收窄 (Union types, narrowing)</strong>：TypeScript 强大的联合类型让数据建模极其灵活，而在 Go 中，这不得不退化为接口或带有大量字段的结构体。</li>
<li><strong>泛型方法与三元运算符</strong>：这些“现代化”特性的缺失，让从前端背景转过来的工程师们颇感不适。</li>
</ul>
<p>然而，对于编译器团队来说，<strong>为了性能，这一切“阵痛”都是值得的</strong>。他们用语法的繁琐换取了运行时的极速，这正是工程世界中最经典的“等价交换”。</p>
<blockquote>
<p>注：<a href="https://tonybai.com/2026/01/24/go-generics-finally-supports-generic-methods">关于泛型方法，Go团队很大可能将在Go 1.27支持！</a></p>
</blockquote>
<h2>未来展望：TypeScript 7.0</h2>
<p>目前，Go 版本的编译器已经能通过 10 万个测试用例，并在 Slack、Figma 等大厂的内部构建中试运行（Slack 的构建时间从 6 分钟降至 40 秒）。</p>
<p>Microsoft 计划在 TypeScript 6.0 中开始引入一些破坏性变更，为 Go 版本的上位做铺垫。而那个完全由 Go 驱动、极速的编译器，预计将被命名为 <strong>TypeScript 7.0</strong>。</p>
<p>这场从 Node.js 到 Go 的大迁徙，不仅证明了 Go 在复杂编译器领域的工程能力，也为所有面临类似性能瓶颈的团队，提供了一个极具参考价值的范本。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2026/typescript-compiler-go-rewrite-10x-speed-microsoft-details-10.png" alt="" /></p>
<blockquote>
<p>注：微软在2025年12月初发布了TypeScript 7.0的最新进展，大家可以在 https://devblogs.microsoft.com/typescript/progress-on-typescript-7-december-2025/ 这里了解详情。</p>
</blockquote>
<p>资料链接：https://www.youtube.com/watch?v=PZm_YbE3fcA</p>
<hr />
<p><strong>你的“重写”冲动</strong></p>
<p>微软用 Go 重写 TS 编译器，是一次壮士断腕般的成功尝试。<strong>在你维护的项目中，是否有那个让你想要“推倒重来”的性能瓶颈？如果让你选，你会<br />
用 Go 还是 Rust 来重写它？</strong></p>
<p><strong>欢迎在评论区分享你的重构经历或选型思考！</strong> 让我们一起探讨如何在性能与开发效率之间找到平衡。</p>
<p><strong>如果这篇文章让你对 Go 在大型项目中的潜力有了新的认识，别忘了点个【赞】和【在看】，并转发给你的架构师朋友！</strong></p>
<hr />
<p>还在为“复制粘贴喂AI”而烦恼？我的新专栏 <strong>《<a href="http://gk.link/a/12EPd">AI原生开发工作流实战</a>》</strong> 将带你：</p>
<ul>
<li>告别低效，重塑开发范式</li>
<li>驾驭AI Agent(Claude Code)，实现工作流自动化</li>
<li>从“AI使用者”进化为规范驱动开发的“工作流指挥家”</li>
</ul>
<p>扫描下方二维码，开启你的AI原生开发之旅。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/ai-native-dev-workflow-qr.png" alt="" /></p>
<hr />
<p>你的Go技能，是否也卡在了“熟练”到“精通”的瓶颈期？</p>
<ul>
<li>想写出更地道、更健壮的Go代码，却总在细节上踩坑？</li>
<li>渴望提升软件设计能力，驾驭复杂Go项目却缺乏章法？</li>
<li>想打造生产级的Go服务，却在工程化实践中屡屡受挫？</li>
</ul>
<p>继《<a href="http://gk.link/a/10AVZ">Go语言第一课</a>》后，我的《<a href="http://gk.link/a/12yGY">Go语言进阶课</a>》终于在极客时间与大家见面了！</p>
<p>我的全新极客时间专栏 《<a href="http://gk.link/a/12yGY">Tony Bai·Go语言进阶课</a>》就是为这样的你量身打造！30+讲硬核内容，带你夯实语法认知，提升设计思维，锻造工程实践能力，更有实战项目串讲。</p>
<p>目标只有一个：助你完成从“Go熟练工”到“Go专家”的蜕变！ 现在就加入，让你的Go技能再上一个新台阶！</p>
<p><img src="https://tonybai.com/wp-content/uploads/course-card/iamtonybai-banner-2.gif" alt="" /></p>
<hr />
<p>商务合作方式：撰稿、出书、培训、在线课程、合伙创业、咨询、广告合作。如有需求，请扫描下方公众号二维码，与我私信联系。</p>
<p><img src="http://image.tonybai.com/img/tonybai/iamtonybai-wechat-qr.png" alt="" /></p>
<p style='text-align:left'>&copy; 2026, <a href='https://tonybai.com'>bigwhite</a>. 版权所有. </p>
]]></content:encoded>
			<wfw:commentRss>https://tonybai.com/2026/01/27/typescript-compiler-go-rewrite-10x-speed-microsoft-details/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>Go 的“浮点数陷阱”将被填平：浮点转整数即将在所有平台上行为一致</title>
		<link>https://tonybai.com/2026/01/11/proposal-float-to-int-conversions-should-saturate-on-overflow/</link>
		<comments>https://tonybai.com/2026/01/11/proposal-float-to-int-conversions-should-saturate-on-overflow/#comments</comments>
		<pubDate>Sat, 10 Jan 2026 23:31:45 +0000</pubDate>
		<dc:creator>bigwhite</dc:creator>
				<category><![CDATA[技术志]]></category>
		<category><![CDATA[amd64]]></category>
		<category><![CDATA[arm64]]></category>
		<category><![CDATA[Compiler]]></category>
		<category><![CDATA[consistency]]></category>
		<category><![CDATA[Conversion]]></category>
		<category><![CDATA[CrossPlatform]]></category>
		<category><![CDATA[DavidChase]]></category>
		<category><![CDATA[exception]]></category>
		<category><![CDATA[FloatingPoint]]></category>
		<category><![CDATA[Go]]></category>
		<category><![CDATA[go1.26]]></category>
		<category><![CDATA[Go1.27]]></category>
		<category><![CDATA[Go1.28]]></category>
		<category><![CDATA[GOEXPERIMENT]]></category>
		<category><![CDATA[Golang]]></category>
		<category><![CDATA[HardwareDifference]]></category>
		<category><![CDATA[IanLanceTaylor]]></category>
		<category><![CDATA[ImplementationDefined]]></category>
		<category><![CDATA[InstructionSet]]></category>
		<category><![CDATA[Integer]]></category>
		<category><![CDATA[NaN]]></category>
		<category><![CDATA[Overflow]]></category>
		<category><![CDATA[PerfectPortability]]></category>
		<category><![CDATA[Performance]]></category>
		<category><![CDATA[Portability]]></category>
		<category><![CDATA[Programmer]]></category>
		<category><![CDATA[RISCV]]></category>
		<category><![CDATA[SaturatingConversion]]></category>
		<category><![CDATA[一致性]]></category>
		<category><![CDATA[可移植性]]></category>
		<category><![CDATA[完美可移植性]]></category>
		<category><![CDATA[实现定义]]></category>
		<category><![CDATA[异常]]></category>
		<category><![CDATA[性能]]></category>
		<category><![CDATA[指令集]]></category>
		<category><![CDATA[整数]]></category>
		<category><![CDATA[浮点数]]></category>
		<category><![CDATA[溢出]]></category>
		<category><![CDATA[硬件差异]]></category>
		<category><![CDATA[程序员]]></category>
		<category><![CDATA[编译器]]></category>
		<category><![CDATA[跨平台]]></category>
		<category><![CDATA[转换]]></category>
		<category><![CDATA[饱和转换]]></category>

		<guid isPermaLink="false">https://tonybai.com/?p=5703</guid>
		<description><![CDATA[本文永久链接 &#8211; https://tonybai.com/2026/01/11/proposal-float-to-int-conversions-should-saturate-on-overflow 大家好，我是Tony Bai。 你是否知道，同一行简单的代码 int64(myFloat)，在 Intel (amd64) 机器上可能返回一个巨大的负数，而在 ARM64 机器上却可能返回最大正整数？ 在 Go 语言中，浮点数到整数的转换溢出行为长期以来一直属于“实现定义”(implementation-dependent) 的灰色地带。这意味着，代码的运行结果竟然取决于你底层的 CPU 架构。这种不确定性，一直是跨平台开发中一个难以察觉的隐形地雷。 2025年末，Go 编译器团队核心成员 David Chase 提交了一份提案（#76264），旨在彻底终结这种混乱。该提案计划在未来的 Go 版本中，强制规定所有平台上的浮点转整数必须是“饱和”的 (saturating)，从而实现真正的全平台行为一致。 痛点：薛定谔的转换结果 在现有的 Go 规范下，如果你尝试将一个超出目标整数范围的浮点数（例如 1e100）转换为 int64，结果是未定义的。 让我们看看这有多疯狂。假设我们有以下代码： var f float64 = 1e100 // 一个巨大的数 var i int64 = int64(f) fmt.Println(i) 这段代码在不同架构下的运行结果截然不同： ARM64, RISC-V: 返回 9223372036854775807 (MAX_INT64)。这是“饱和”行为，即卡在最大值。 AMD64 (x86-64): 返回 [...]]]></description>
			<content:encoded><![CDATA[<p><img src="https://tonybai.com/wp-content/uploads/2026/proposal-float-to-int-conversions-should-saturate-on-overflow-1.png" alt="" /></p>
<p><a href="https://tonybai.com/2026/01/11/proposal-float-to-int-conversions-should-saturate-on-overflow">本文永久链接</a> &#8211; https://tonybai.com/2026/01/11/proposal-float-to-int-conversions-should-saturate-on-overflow</p>
<p>大家好，我是Tony Bai。</p>
<p>你是否知道，同一行简单的代码 int64(myFloat)，在 Intel (amd64) 机器上可能返回一个巨大的负数，而在 ARM64 机器上却可能返回最大正整数？</p>
<p>在 Go 语言中，浮点数到整数的转换溢出行为长期以来一直属于“实现定义”(implementation-dependent) 的灰色地带。这意味着，代码的运行结果竟然取决于你底层的 CPU 架构。这种不确定性，一直是跨平台开发中一个难以察觉的隐形地雷。</p>
<p>2025年末，Go 编译器团队核心成员 David Chase 提交了一份提案（<a href="https://github.com/golang/go/issues/76264">#76264</a>），旨在彻底终结这种混乱。该提案计划在未来的 Go 版本中，<strong>强制规定所有平台上的浮点转整数必须是“饱和”的 (saturating)</strong>，从而实现真正的全平台行为一致。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2026/distributed-system-guide-qr.png" alt="img{512x368}" /></p>
<h2>痛点：薛定谔的转换结果</h2>
<p>在现有的 Go 规范下，如果你尝试将一个超出目标整数范围的浮点数（例如 1e100）转换为 int64，结果是未定义的。</p>
<p>让我们看看这有多疯狂。假设我们有以下代码：</p>
<pre><code class="go">var f float64 = 1e100 // 一个巨大的数
var i int64 = int64(f)
fmt.Println(i)
</code></pre>
<p>这段代码在不同架构下的运行结果截然不同：</p>
<ul>
<li><strong>ARM64, RISC-V</strong>: 返回 9223372036854775807 (<strong>MAX_INT64</strong>)。这是“饱和”行为，即卡在最大值。</li>
<li><strong>AMD64 (x86-64)</strong>: 返回 -9223372036854775808 (<strong>MIN_INT64</strong>)。这是一个令人困惑的溢出结果。</li>
<li><strong>WASM</strong>: 行为又不一样&#8230;</li>
</ul>
<p>更糟糕的是 NaN (Not a Number) 的转换：</p>
<pre><code class="go">var j int64 = int64(math.NaN())
fmt.Println(j)
</code></pre>
<ul>
<li><strong>ARM64</strong>: 返回 0。</li>
<li><strong>AMD64</strong>: 返回 <strong>MIN_INT64</strong>。</li>
<li><strong>RISC-V</strong>: 返回 <strong>MAX_INT64</strong>。</li>
</ul>
<p>这种不一致性不仅仅是理论问题，它已经导致了准标准库 x/time/rate 中的真实 Bug (<a href="https://github.com/golang/go/issues/71154">#71154</a>)。当你的代码逻辑依赖于转换结果的正负号来做判断时（例如 if i > 0），这种硬件差异就是致命的。</p>
<h2>解决方案：拥抱“饱和转换”</h2>
<p>David Chase 的提案非常直接：<strong>统一行为，拥抱饱和。</strong></p>
<p>所谓“饱和转换”，是指当浮点数超出目标整数的表示范围时，结果应该被“钳制”在目标类型的最大值或最小值，而不是发生回绕(wraparound)或产生随机值。</p>
<p>具体规则如下：</p>
<ol>
<li><strong>正溢出</strong> -> 返回目标类型的 <strong>最大值</strong> (MaxInt)。</li>
<li><strong>负溢出</strong> -> 返回目标类型的 <strong>最小值</strong> (MinInt)。</li>
<li><strong>NaN</strong> -> 返回 <strong>0</strong> (或归一化为 0)。</li>
</ol>
<p>这一改变将使得 Go 代码在任何 CPU 架构上都表现出完全一致的逻辑，彻底消除了这类可移植性隐患。</p>
<h2>深层权衡：一致性 vs. 性能</h2>
<p>为什么 Go 以前不这么做？核心原因在于<strong>性能成本</strong>。</p>
<p>在 ARM64 和 RISC-V 等现代架构上，硬件指令集（如 FCVT）原生支持饱和转换，因此这样做几乎没有额外开销。</p>
<p>然而，<strong>AMD64 (x86-64) 是个“异类”</strong>。它的 CVTTSD2SQ 指令在溢出时不仅返回一个特殊的“不定值”（通常是 MinInt），还会触发浮点异常。为了在 AMD64 上模拟出“饱和”行为，编译器必须插入额外的检查代码：</p>
<pre><code class="go">// 模拟代码逻辑：AMD64 上的额外开销
result = int64(x)
if result == MIN_INT64 { // 可能溢出了
    if x &gt; 0 {
        result = MAX_INT64 // 正溢出修正
    } else if !(x &lt; 0) {
        result = 0         // NaN 修正
    }
}
</code></pre>
<p>Go 核心团队成员 Ian Lance Taylor 在评论中指出，我们必须权衡：<strong>为了消除这种不一致性，值得让 AMD64 上的转换操作变慢吗？</strong></p>
<p>提案作者 David Chase 的回应是：<strong>值得。</strong> 与 FMA (融合乘加) 指令带来的微小精度差异不同，浮点转整数的差异往往是<strong>正负号级别</strong>的（MaxInt vs MinInt），这直接决定了代码逻辑的走向（循环是否执行、条件是否满足）。这种差异带来的 Bug 极其隐蔽且难以调试，其代价远超那几条指令的性能损耗。</p>
<h2>实施计划：温和的演进</h2>
<p>为了避免生态系统的剧烈震荡，提案建议采用分阶段的落地策略：</p>
<ul>
<li><strong>Go 1.26</strong>: 引入 GOEXPERIMENT 标志，允许开发者尝鲜并测试影响。</li>
<li><strong>Go 1.27</strong>: 将其设为默认的实现行为。</li>
<li><strong>Go 1.28</strong>: 正式修改 Go 语言规范 (Spec)，将其确立为标准。</li>
</ul>
<blockquote>
<p>注：Go 1.26当前已经功能冻结，<a href="https://github.com/golang/go/issues/33892#issuecomment-3721268260">该提案依然处于Go语言规范变更审查委员会的讨论状态中</a>，因此即便逻辑，其实际落地时间表也会顺延。</p>
</blockquote>
<h2>小结：Go 向“完美可移植性”迈出的重要一步</h2>
<p>Dr Chase的这个提案不仅是对一个技术细节的修正，更是 Go 语言设计哲学的一次体现：<strong>在工程实践中，可预测性和可移植性往往优于特定平台上的极致微优化。</strong></p>
<p>如果该提案通过，未来的 Gopher 们将不再需要担心底层的 CPU 是 Intel 还是 ARM，int64(NaN) 永远是 0，int64(Inf) 永远是 MaxInt64。这，才是我们想要的“Write Once, Run Anywhere”。</p>
<blockquote>
<p>注：目前Dr Chase也在努力弥合amd64下的性能差距。</p>
</blockquote>
<p>资料链接：https://github.com/golang/go/issues/76264</p>
<hr />
<p><strong>你的跨平台“血泪史”</strong></p>
<p>跨平台开发中的“未定义行为”往往是最难调试的 Bug。<strong>在你的开发生涯中，是否也遇到过因为 CPU 架构或操作系统差异而导致的诡异问题？你支持为了“一致性”而牺牲一点点 AMD64 上的性能吗？</strong></p>
<p><strong>欢迎在评论区分享你的踩坑经历或对提案的看法！</strong> 让我们一起见证 Go 语言的进化。</p>
<p><strong>如果这篇文章让你对底层原理有了新的认识，别忘了点个【赞】和【在看】，并转发给你的硬核伙伴！</strong></p>
<hr />
<p>还在为“复制粘贴喂AI”而烦恼？我的新专栏 <strong>《<a href="http://gk.link/a/12EPd">AI原生开发工作流实战</a>》</strong> 将带你：</p>
<ul>
<li>告别低效，重塑开发范式</li>
<li>驾驭AI Agent(Claude Code)，实现工作流自动化</li>
<li>从“AI使用者”进化为规范驱动开发的“工作流指挥家”</li>
</ul>
<p>扫描下方二维码，开启你的AI原生开发之旅。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/ai-native-dev-workflow-qr.png" alt="" /></p>
<hr />
<p>你的Go技能，是否也卡在了“熟练”到“精通”的瓶颈期？</p>
<ul>
<li>想写出更地道、更健壮的Go代码，却总在细节上踩坑？</li>
<li>渴望提升软件设计能力，驾驭复杂Go项目却缺乏章法？</li>
<li>想打造生产级的Go服务，却在工程化实践中屡屡受挫？</li>
</ul>
<p>继《<a href="http://gk.link/a/10AVZ">Go语言第一课</a>》后，我的《<a href="http://gk.link/a/12yGY">Go语言进阶课</a>》终于在极客时间与大家见面了！</p>
<p>我的全新极客时间专栏 《<a href="http://gk.link/a/12yGY">Tony Bai·Go语言进阶课</a>》就是为这样的你量身打造！30+讲硬核内容，带你夯实语法认知，提升设计思维，锻造工程实践能力，更有实战项目串讲。</p>
<p>目标只有一个：助你完成从“Go熟练工”到“Go专家”的蜕变！ 现在就加入，让你的Go技能再上一个新台阶！</p>
<p><img src="https://tonybai.com/wp-content/uploads/course-card/iamtonybai-banner-2.gif" alt="" /></p>
<hr />
<p>商务合作方式：撰稿、出书、培训、在线课程、合伙创业、咨询、广告合作。如有需求，请扫描下方公众号二维码，与我私信联系。</p>
<p><img src="http://image.tonybai.com/img/tonybai/iamtonybai-wechat-qr.png" alt="" /></p>
<p style='text-align:left'>&copy; 2026, <a href='https://tonybai.com'>bigwhite</a>. 版权所有. </p>
]]></content:encoded>
			<wfw:commentRss>https://tonybai.com/2026/01/11/proposal-float-to-int-conversions-should-saturate-on-overflow/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>让编译器成为你的副驾驶：告别“防御性编程”，拥抱“类型驱动开发”</title>
		<link>https://tonybai.com/2026/01/04/stop-lying-to-the-compiler/</link>
		<comments>https://tonybai.com/2026/01/04/stop-lying-to-the-compiler/#comments</comments>
		<pubDate>Sun, 04 Jan 2026 05:27:08 +0000</pubDate>
		<dc:creator>bigwhite</dc:creator>
				<category><![CDATA[技术志]]></category>
		<category><![CDATA[any]]></category>
		<category><![CDATA[CodeQuality]]></category>
		<category><![CDATA[Compiler]]></category>
		<category><![CDATA[CompileTimeSafety]]></category>
		<category><![CDATA[DefensiveProgramming]]></category>
		<category><![CDATA[Go]]></category>
		<category><![CDATA[Golang]]></category>
		<category><![CDATA[IllegalState]]></category>
		<category><![CDATA[Interface]]></category>
		<category><![CDATA[NewPort]]></category>
		<category><![CDATA[nil]]></category>
		<category><![CDATA[nil指针]]></category>
		<category><![CDATA[Optionality]]></category>
		<category><![CDATA[OrderID]]></category>
		<category><![CDATA[panic]]></category>
		<category><![CDATA[RuntimeErrors]]></category>
		<category><![CDATA[TypeAssertions]]></category>
		<category><![CDATA[TypeDrivenDevelopment]]></category>
		<category><![CDATA[TypeSystem]]></category>
		<category><![CDATA[UserID]]></category>
		<category><![CDATA[代码质量]]></category>
		<category><![CDATA[可空性]]></category>
		<category><![CDATA[构造函数]]></category>
		<category><![CDATA[类型断言]]></category>
		<category><![CDATA[类型系统]]></category>
		<category><![CDATA[类型驱动开发]]></category>
		<category><![CDATA[编译器]]></category>
		<category><![CDATA[编译时安全]]></category>
		<category><![CDATA[运行时错误]]></category>
		<category><![CDATA[防御性编程]]></category>
		<category><![CDATA[非法状态]]></category>

		<guid isPermaLink="false">https://tonybai.com/?p=5663</guid>
		<description><![CDATA[本文永久链接 &#8211; https://tonybai.com/2026/01/04/stop-lying-to-the-compiler 大家好，我是Tony Bai。 “半夜被值班的运维同事叫醒，发现生产环境崩了，原因是一个深藏在业务逻辑里的 nil 指针异常。” 这个场景，对于每个后端开发者来说都是挥之不去的噩梦。事后复盘时，我们往往会懊恼：“为什么这里没加 if != nil 判断？”然后，我们在代码里撒上一把防御性检查的“盐”，祈祷下次好运。 但这真的是解决之道吗？ 最近，Daniel Beskin 的一篇深度好文《The Compiler Is Your Best Friend, Stop Lying to It》（编译器是你最好的朋友，别再对它撒谎了），为我们提供了一个全新的视角：这些运行时崩溃，本质上是因为我们在编译时对编译器撒了谎。 我们告诉编译器“这是一个字符串”，但实际上它可能是 nil；我们告诉编译器“这个函数返回一个整数”，但实际上它可能抛出一个 panic。当我们停止撒谎，开始用类型系统表达真实意图时，编译器将从一个“报错机器”，变成我们最强大的“安全副驾驶”。 我们对编译器撒过的“谎” 在 Go 语言的日常开发中，我们常常为了“方便”而向编译器撒谎，埋下了日后爆炸的地雷。 谎言一：隐形的 nil 当我们定义 func Process(u *User) 时，我们告诉编译器：“给我一个 User，我处理它。” 但在 Go 中，指针可以是 nil。 * 谎言：我承诺会处理一个 User。 * 真相：我可能会收到一个 nil，然后炸掉。 * 后果：为了弥补这个谎言，我们需要在函数内部写无数的 if u [...]]]></description>
			<content:encoded><![CDATA[<p><img src="https://tonybai.com/wp-content/uploads/2026/stop-lying-to-the-compiler-1.png" alt="" /></p>
<p><a href="https://tonybai.com/2026/01/04/stop-lying-to-the-compiler">本文永久链接</a> &#8211; https://tonybai.com/2026/01/04/stop-lying-to-the-compiler</p>
<p>大家好，我是Tony Bai。</p>
<p>“半夜被值班的运维同事叫醒，发现生产环境崩了，原因是一个深藏在业务逻辑里的 nil 指针异常。”</p>
<p>这个场景，对于每个后端开发者来说都是挥之不去的噩梦。事后复盘时，我们往往会懊恼：“为什么这里没加 if != nil 判断？”然后，我们在代码里撒上一把防御性检查的“盐”，祈祷下次好运。</p>
<p>但这真的是解决之道吗？</p>
<p>最近，Daniel Beskin 的一篇深度好文《<a href="https://blog.daniel-beskin.com/2025-12-22-the-compiler-is-your-best-friend-stop-lying-to-it">The Compiler Is Your Best Friend, Stop Lying to It</a>》（编译器是你最好的朋友，别再对它撒谎了），为我们提供了一个全新的视角：<strong>这些运行时崩溃，本质上是因为我们在编译时对编译器撒了谎。</strong></p>
<p>我们告诉编译器“这是一个字符串”，但实际上它可能是 nil；我们告诉编译器“这个函数返回一个整数”，但实际上它可能抛出一个 panic。当我们停止撒谎，开始用类型系统表达真实意图时，编译器将从一个“报错机器”，变成我们最强大的“安全副驾驶”。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/paid/go-micro-column-2025-pr.png" alt="" /></p>
<h2>我们对编译器撒过的“谎”</h2>
<p>在 Go 语言的日常开发中，我们常常为了“方便”而向编译器撒谎，埋下了日后爆炸的地雷。</p>
<h3>谎言一：隐形的 nil</h3>
<p>当我们定义 func Process(u *User) 时，我们告诉编译器：“给我一个 User，我处理它。” 但在 Go 中，指针可以是 nil。<br />
*   <strong>谎言</strong>：我承诺会处理一个 User。<br />
*   <strong>真相</strong>：我可能会收到一个 nil，然后炸掉。<br />
*   <strong>后果</strong>：为了弥补这个谎言，我们需要在函数内部写无数的 if u == nil 防御性代码。一旦遗漏，就是生产事故。</p>
<h3>谎言二：盲目的类型断言与 any</h3>
<p>当我们使用 interface{} (或 any) 时，我们实际上是在对编译器说：“别管这个，我知道我在做什么。”<br />
*   <strong>谎言</strong>：这个 any 类型的变量，其实是一个 int。<br />
*   <strong>真相</strong>：它可能是一个 string，或者 nil。<br />
*   <strong>后果</strong>：运行时的 panic: interface conversion: interface {} is string, not int。</p>
<h3>谎言三：隐藏的副作用与 Panic</h3>
<p>当我们看到一个函数签名 func Parse(s string) int 时，编译器认为它是一个将字符串映射为整数的函数。<br />
*   <strong>谎言</strong>：这是一个纯粹的转换函数。<br />
*   <strong>真相</strong>：如果字符串格式不对，我会直接 panic，中断整个 goroutine。<br />
*   <strong>后果</strong>：调用者无法通过函数签名预知风险，导致程序在边缘情况下意外崩溃。</p>
<h2>停止撒谎，开启“对话”</h2>
<p>如何重建与编译器的信任关系？答案是：<strong>将运行时的检查，提前到编译时的类型定义中。</strong></p>
<h3>策略一：让非法状态无法表示</h3>
<p>这是消除 nil 和无效数据的终极心法。</p>
<ul>
<li><strong>场景</strong>：一个配置项 Port，如果是 0 表示随机端口，如果是正数表示指定端口。</li>
<li><strong>糟糕的设计</strong>：Port int。你必须在代码各处检查 Port &lt; 0 的情况，并且含义模糊。</li>
<li>
<p><strong>诚实的设计</strong>：</p>
<pre><code class="go">type Port int

// 使用构造函数来保证 Port 的合法性
func NewPort(p int) (Port, error) {
    if p &lt; 0 || p &gt; 65535 {
        return 0, fmt.Errorf("invalid port")
    }
    return Port(p), nil
}
</code></pre>
<p>一旦你通过 NewPort 拥有了一个 Port 类型的值，编译器就为你担保：<strong>它一定是一个合法的端口号</strong>。你后续不再需要防御性检查(未通过NewPort获得的除外)。</p>
</li>
</ul>
<h3>策略二：用类型区分概念</h3>
<ul>
<li><strong>场景</strong>：用户 ID 和 订单 ID 都是 int64。</li>
<li><strong>糟糕的设计</strong>：func GetOrder(userID, orderID int64)。调用者很容易把两个 ID 传反，而编译器毫无察觉。</li>
<li>
<p><strong>诚实的设计</strong>：</p>
<pre><code class="go">type UserID int64
type OrderID int64

func GetOrder(uid UserID, oid OrderID) { ... }
</code></pre>
<p>现在，如果你试图把 UserID 传给 OrderID，编译器会直接报错。这不是繁琐，这是<strong>编译器在帮你 Review 代码</strong>。</p>
</li>
</ul>
<h3>策略三：显式的可空性</h3>
<p>虽然 Go 没有 Rust 的 Option<T>，但我们可以利用指针的语义来诚实地表达“可能不存在”。</p>
<ul>
<li><strong>场景</strong>：更新用户信息，只更新非空字段。</li>
<li><strong>诚实的设计</strong>：<br />
<code>go<br />
type UpdateUserRequest struct {<br />
    Name *string // nil 表示不更新，非 nil 表示更新为新值<br />
    Age  *int<br />
}</code><br />
这里，指针不再是“可能导致崩溃的引用”，而是<strong>“可选值”的显式类型标记</strong>。这让代码的意图对编译器和人类都一目了然。</li>
</ul>
<h2>编译器是你的朋友，不是敌人</h2>
<p>很多时候，我们觉得编译器很烦人：它阻止我们快速写出“能跑”的代码，强迫我们处理每一个 err，纠结于类型转换。</p>
<p>但 Daniel Beskin 提醒我们：<strong>编译器是你唯一一个会不厌其烦地帮你检查每一个细节、永远不会疲倦、永远不会因为“差不多就行”而放过 Bug 的队友。</strong></p>
<p>当你觉得编译器在“阻碍”你时，停下来想一想：<strong>是不是我在试图对它撒谎？</strong></p>
<ul>
<li>如果类型不匹配，是不是我的数据模型设计得不够清晰？</li>
<li>如果错误处理太繁琐，是不是因为我试图把不确定的状态传递得太远？</li>
</ul>
<h2>小结：睡个好觉的秘诀</h2>
<p>“防御性编程”是一种补救措施，它假设代码是脆弱的。而“类型驱动开发”是一种预防措施，它利用编译器构建坚固的堡垒。</p>
<p>当我们开始<strong>尊重类型</strong>，停止用 any 和隐式约定来糊弄编译器时，我们获得的回报是巨大的：</p>
<ul>
<li><strong>重构时的自信</strong>：修改一个类型，编译器会告诉你所有需要调整的地方。</li>
<li><strong>更少的测试</strong>：你不需要测试“端口号是否为负数”，因为类型系统保证了它不可能为负。</li>
<li><strong>更安稳的睡眠</strong>：因为你知道，那些导致半夜崩溃的低级错误，早在你按下 go build 的那一刻，就被忠诚的编译器拦截在了门外。</li>
</ul>
<p>资料链接：https://blog.daniel-beskin.com/2025-12-22-the-compiler-is-your-best-friend-stop-lying-to-it</p>
<hr />
<p><strong>你的“撒谎”时刻</strong></p>
<p>读完这篇文章，你是否也意识到了自己曾在代码中对编译器撒过的“谎”？<strong>在你的项目中，有哪些因为类型定义不清而导致的“血案”？或者，你有哪些利用类型系统来规避 Bug 的独门绝技？</strong></p>
<p><strong>欢迎在评论区分享你的反思与心得！</strong> 让我们一起学会“诚实”编程，睡个好觉。</p>
<p><strong>如果这篇文章颠覆了你对编译器的认知，别忘了点个【赞】和【在看】，并转发给你的团队，一起提升代码的“诚实度”！</strong></p>
<hr />
<p>还在为“复制粘贴喂AI”而烦恼？我的新专栏 <strong>《<a href="http://gk.link/a/12EPd">AI原生开发工作流实战</a>》</strong> 将带你：</p>
<ul>
<li>告别低效，重塑开发范式</li>
<li>驾驭AI Agent(Claude Code)，实现工作流自动化</li>
<li>从“AI使用者”进化为规范驱动开发的“工作流指挥家”</li>
</ul>
<p>扫描下方二维码，开启你的AI原生开发之旅。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/ai-native-dev-workflow-qr.png" alt="" /></p>
<hr />
<p>你的Go技能，是否也卡在了“熟练”到“精通”的瓶颈期？</p>
<ul>
<li>想写出更地道、更健壮的Go代码，却总在细节上踩坑？</li>
<li>渴望提升软件设计能力，驾驭复杂Go项目却缺乏章法？</li>
<li>想打造生产级的Go服务，却在工程化实践中屡屡受挫？</li>
</ul>
<p>继《<a href="http://gk.link/a/10AVZ">Go语言第一课</a>》后，我的《<a href="http://gk.link/a/12yGY">Go语言进阶课</a>》终于在极客时间与大家见面了！</p>
<p>我的全新极客时间专栏 《<a href="http://gk.link/a/12yGY">Tony Bai·Go语言进阶课</a>》就是为这样的你量身打造！30+讲硬核内容，带你夯实语法认知，提升设计思维，锻造工程实践能力，更有实战项目串讲。</p>
<p>目标只有一个：助你完成从“Go熟练工”到“Go专家”的蜕变！ 现在就加入，让你的Go技能再上一个新台阶！</p>
<p><img src="https://tonybai.com/wp-content/uploads/course-card/iamtonybai-banner-2.gif" alt="" /></p>
<hr />
<p>商务合作方式：撰稿、出书、培训、在线课程、合伙创业、咨询、广告合作。如有需求，请扫描下方公众号二维码，与我私信联系。</p>
<p><img src="http://image.tonybai.com/img/tonybai/iamtonybai-wechat-qr.png" alt="" /></p>
<p style='text-align:left'>&copy; 2026, <a href='https://tonybai.com'>bigwhite</a>. 版权所有. </p>
]]></content:encoded>
			<wfw:commentRss>https://tonybai.com/2026/01/04/stop-lying-to-the-compiler/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>Go 2026 路线图曝光：SIMD、泛型方法与无 C 工具链 CGO —— 性能与表达力的双重飞跃？</title>
		<link>https://tonybai.com/2025/11/28/go-2026-roadmap-revealed/</link>
		<comments>https://tonybai.com/2025/11/28/go-2026-roadmap-revealed/#comments</comments>
		<pubDate>Fri, 28 Nov 2025 00:45:22 +0000</pubDate>
		<dc:creator>bigwhite</dc:creator>
				<category><![CDATA[技术志]]></category>
		<category><![CDATA[arena]]></category>
		<category><![CDATA[arm64]]></category>
		<category><![CDATA[async/await]]></category>
		<category><![CDATA[AVX512]]></category>
		<category><![CDATA[CacheLineContention]]></category>
		<category><![CDATA[Cgo]]></category>
		<category><![CDATA[Clang]]></category>
		<category><![CDATA[GC]]></category>
		<category><![CDATA[GCC]]></category>
		<category><![CDATA[genericmethod]]></category>
		<category><![CDATA[Go]]></category>
		<category><![CDATA[go1.26]]></category>
		<category><![CDATA[Golang]]></category>
		<category><![CDATA[goroutine]]></category>
		<category><![CDATA[highlevelAPI]]></category>
		<category><![CDATA[JavaScript]]></category>
		<category><![CDATA[L1/L2Cache]]></category>
		<category><![CDATA[LastLevelCache]]></category>
		<category><![CDATA[LLC]]></category>
		<category><![CDATA[memoryregions]]></category>
		<category><![CDATA[NEON]]></category>
		<category><![CDATA[NUMA]]></category>
		<category><![CDATA[Promise]]></category>
		<category><![CDATA[purego]]></category>
		<category><![CDATA[runtime.free]]></category>
		<category><![CDATA[runtime.freegc]]></category>
		<category><![CDATA[scalablevectors]]></category>
		<category><![CDATA[Schedulingaffinity]]></category>
		<category><![CDATA[Shardedvalues]]></category>
		<category><![CDATA[SIMD]]></category>
		<category><![CDATA[Specializedmalloc]]></category>
		<category><![CDATA[strings.Builder]]></category>
		<category><![CDATA[STW]]></category>
		<category><![CDATA[SVE]]></category>
		<category><![CDATA[sync.Sharded]]></category>
		<category><![CDATA[Tensor]]></category>
		<category><![CDATA[uniontype]]></category>
		<category><![CDATA[wasm]]></category>
		<category><![CDATA[Wasmstackswitching]]></category>
		<category><![CDATA[WriteBarrier]]></category>
		<category><![CDATA[云原生]]></category>
		<category><![CDATA[交叉编译]]></category>
		<category><![CDATA[内存区域]]></category>
		<category><![CDATA[内存释放]]></category>
		<category><![CDATA[分片值]]></category>
		<category><![CDATA[单指令多数据]]></category>
		<category><![CDATA[向量化算法]]></category>
		<category><![CDATA[垃圾回收]]></category>
		<category><![CDATA[性能]]></category>
		<category><![CDATA[无C工具链CGO]]></category>
		<category><![CDATA[显式释放]]></category>
		<category><![CDATA[模式匹配]]></category>
		<category><![CDATA[泛型方法]]></category>
		<category><![CDATA[编译器]]></category>
		<category><![CDATA[联合类型]]></category>
		<category><![CDATA[表达力]]></category>
		<category><![CDATA[调度亲和性]]></category>
		<category><![CDATA[路线图]]></category>
		<category><![CDATA[运行时]]></category>
		<category><![CDATA[逃逸分析]]></category>

		<guid isPermaLink="false">https://tonybai.com/?p=5450</guid>
		<description><![CDATA[本文永久链接 &#8211; https://tonybai.com/2025/11/28/go-2026-roadmap-revealed 大家好，我是Tony Bai。 在最近的一期 Go 编译器与运行时团队会议纪要中，我们惊喜地发现了一份关于 2026 年的规划 (2026 planning，如下图)。这份规划虽然简短，但其包含的信息量却足以让任何一位关注 Go 语言未来的开发者心跳加速。 从榨干硬件潜能的 SIMD 和运行时手动内存释放(runtime.free)，到呼声极高的泛型方法(generic method)与联合类型(union type)，再到彻底解决交叉编译痛点的无 C 工具链 CGO，Go 团队正密谋着一场关于性能、表达力与工程体验的全方位变革。 本文将结合最新的设计文档、CL (Change List) 记录和社区核心 Issue，和大家一起解析一下这份 Go 2026 路线图背后的技术细节与战略意图。 性能的极限突围 —— 榨干硬件的每一滴油水 一直以来，Go 在性能上的策略都是“足够好”。但在 2026 规划中，我们看到了 Go 团队向“极致性能”发起的冲锋，目标直指 AI、科学计算和高频交易等对延迟极度敏感的领域。 SIMD：从“汇编黑魔法”到“原生公民” 关键词：SIMD (ARM64, scalable vectors &#38; high-level API) 解读： 现状：目前在 Go 中使用 SIMD（单指令多数据）主要依赖手写汇编，不仅难以维护，而且无法被编译器内联优化，甚至会阻碍异步抢占。 变革：规划明确提出了 [...]]]></description>
			<content:encoded><![CDATA[<p><img src="https://tonybai.com/wp-content/uploads/2025/go-2026-roadmap-revealed-1.png" alt="" /></p>
<p><a href="https://tonybai.com/2025/11/28/go-2026-roadmap-revealed">本文永久链接</a> &#8211; https://tonybai.com/2025/11/28/go-2026-roadmap-revealed</p>
<p>大家好，我是Tony Bai。</p>
<p>在最近的一期 <a href="https://github.com/golang/go/issues/43930#issuecomment-3576250284">Go 编译器与运行时团队会议纪要</a>中，我们惊喜地发现了一份关于 <strong>2026 年的规划 (2026 planning，如下图)</strong>。这份规划虽然简短，但其包含的信息量却足以让任何一位关注 Go 语言未来的开发者心跳加速。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/go-2026-roadmap-revealed-2.png" alt="" /></p>
<p>从榨干硬件潜能的 <strong><a href="https://tonybai.com/2025/08/22/go-simd-package-preview">SIMD</a></strong> 和<strong>运行时手动内存释放(<a href="https://tonybai.com/2025/09/18/go-runtime-free-proposal">runtime.free</a>)</strong>，到呼声极高的<strong>泛型方法(generic method)</strong>与<strong>联合类型(union type)</strong>，再到彻底解决交叉编译痛点的<strong>无 C 工具链 CGO</strong>，Go 团队正密谋着一场关于<strong>性能、表达力与工程体验</strong>的全方位变革。</p>
<p>本文将结合最新的设计文档、CL (Change List) 记录和社区核心 Issue，和大家一起解析一下这份 Go 2026 路线图背后的技术细节与战略意图。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/paid/google-adk-in-action-qr.png" alt="" /></p>
<hr />
<h2>性能的极限突围 —— 榨干硬件的每一滴油水</h2>
<p>一直以来，Go 在性能上的策略都是“足够好”。但在 2026 规划中，我们看到了 Go 团队向“极致性能”发起的冲锋，目标直指 AI、科学计算和高频交易等对延迟极度敏感的领域。</p>
<h3>SIMD：从“汇编黑魔法”到“原生公民”</h3>
<ul>
<li><strong>关键词</strong>：SIMD (ARM64, scalable vectors &amp; high-level API)</li>
<li><strong>解读</strong>：
<ul>
<li><strong>现状</strong>：目前在 Go 中使用 SIMD（单指令多数据）主要依赖手写汇编，不仅难以维护，而且无法被编译器内联优化，甚至会阻碍异步抢占。</li>
<li><strong>变革</strong>：规划明确提出了 <strong>“high-level API”</strong>。这意味着 Go 将提供一套<strong>原生的、类型安全的 SIMD 库</strong>。开发者可以用纯 Go 代码编写向量化算法，由编译器自动映射到底层的 AVX-512 (x86) 或 NEON/SVE (ARM) 指令。</li>
<li><strong>Scalable Vectors</strong>：特别提到的“可伸缩向量”，直指 ARM64 的 <strong>SVE (Scalable Vector Extension)</strong> 技术。这将允许同一份 Go 二进制代码，在不同向量长度（128位到2048位）的硬件上自动适配，实现性能的“线性扩展”，这对于 AI 推理场景至关重要。</li>
<li><strong>进展</strong>：在2026年初发布的Go 1.26中，Cherry Mui 提交的关于 Architecture-specific SIMD intrinsics 的提案将以GO实验特性落地，这意味着Go开发者将拥有原生的simd包实现，目前这一工作已在紧锣密鼓地进行中。</li>
</ul>
</li>
</ul>
<h3>runtime.free：打破 GC 的“金科玉律”</h3>
<ul>
<li><strong>关键词</strong>：runtime.free, Specialized malloc</li>
<li><strong>解读</strong>：这是一个颠覆性的变化。Go 一直以自动 GC 著称，但在极致性能场景下，GC 的 CPU 和 STW 开销仍是瓶颈。
<ul>
<li><strong>显式释放</strong>：根据设计文档 《<a href="https://go.dev/design/74299-runtime-freegc">Directly freeing user memory to reduce GC work</a> 》和相关 CL (如 CL 673695)，runtime.freegc 允许将不再使用的堆内存<strong>立即归还</strong>给分配器，供后续重用，而<strong>完全绕过 GC 扫描</strong>。</li>
<li><strong>编译器辅助</strong>：这并非让用户手动管理内存（那样太不安全）。Go 的愿景是让<strong>编译器</strong>通过逃逸分析和生命周期分析，<strong>自动插入</strong> free 调用。例如，在 strings.Builder 的扩容过程中，旧的 buffer 可以被立即释放。</li>
<li><strong>实测数据</strong>：在早期的原型测试中，优化后的 strings.Builder 性能提升了 <strong>2 倍</strong>！配合针对无指针对象 (noscan) 优化的专用分配器 (Specialized malloc)，Go 的临时对象分配性能将逼近栈分配。</li>
</ul>
</li>
</ul>
<hr />
<h2>可伸缩性的新高度 —— 拥抱超多核时代</h2>
<p>随着 CPU 核心数向 128 核甚至更高迈进，传统的并发模式开始遇到“扩展性墙”。Go 2026 规划给出了一套组合拳。</p>
<h3>分片值 (Sharded Values)</h3>
<ul>
<li><strong>关键词</strong>：Sharded values</li>
<li><strong>痛点</strong>：在高并发场景下，对同一个全局计数器或 sync.Pool 的访问，会导致严重的<strong>缓存行争用 (Cache Line Contention)</strong>，让多核优势荡然无存。</li>
<li><strong>解决方案</strong>：Go团队提出一个名为<a href="https://tonybai.com/2025/05/19/shardedvalue-per-cpu-proposal/">sync.Sharded</a> 的提案(详见 Issue #18802)，sync.Sharded 旨在提供一种<strong>“每 P (Processor) 本地化”</strong>的数据结构。
<ul>
<li><strong>无锁读写</strong>：每个 P 只操作自己本地的分片，完全无锁，零竞争。</li>
<li><strong>按需聚合</strong>：只在需要读取总值时，才遍历所有分片进行聚合。</li>
<li>这比现有的 sync.Map 或 atomic 操作在高核数机器上将有数量级的性能提升。</li>
</ul>
</li>
</ul>
<h3>调度亲和性 (Scheduling Affinity)</h3>
<ul>
<li><strong>关键词</strong>：Scheduling affinity</li>
<li><strong>解读</strong>：Go 调度器的“工作窃取”机制虽然平衡了负载，但也导致 Goroutine 经常在不同 CPU 核心间“漂移”，破坏了 L1/L2 缓存的热度。
<ul>
<li><strong>新机制</strong>：在 Issue #65694中，Go团队 计划引入一种机制，允许将一组相关的 Goroutine <strong>“绑定”</strong> 或 <strong>“倾向”</strong> 于特定的 P 或 NUMA 节点。这对于数据库、高频交易系统等缓存敏感型应用是巨大的利好，能显著减少 <strong>LLC (Last Level Cache) Miss</strong>。</li>
</ul>
</li>
</ul>
<h3>内存区域 (Memory Regions)</h3>
<ul>
<li><strong>关键词</strong>：Memory regions</li>
<li><strong>解读</strong>：在 <strong>Arena</strong>试验失败后，Michael Knyszek发起了一个名为Memory regions方案的讨论（具体见 <a href="https://github.com/golang/go/discussions/70257">Discussion #70257</a>)，其核心思想是，通过一个 region.Do(func() { &#8230; }) 调用，将一个函数作用域内的所有内存分配<strong>隐式地</strong>绑定到一个临时的、与 goroutine 绑定的区域中。这个优雅设计的背后，是<strong>极其复杂的实现</strong>。它需要在开启区域的 goroutine 中启用一个特殊的、低开销的<strong>写屏障（write barrier）</strong>来动态追踪内存的逃逸。虽然理论上可行，但其实现复杂度和潜在的性能开销，使其成为一个长期且充满不确定性的研究课题。在2026年，Go团队要在这个方案上有所突破，依旧任重道远。</li>
</ul>
<hr />
<h2>语言表达力的觉醒 —— 填补泛型后的最后拼图</h2>
<p>在泛型落地后，Go 社区对语言特性的渴望并未止步。规划中提到的几个特性，将进一步提升 Go 的表达力。</p>
<h3>泛型方法 (Generic Methods)</h3>
<ul>
<li><strong>关键词</strong>：generic methods</li>
<li><strong>背景</strong>：这是泛型引入后最大的遗憾之一。目前 Go 不支持在接口方法或结构体方法中定义额外的类型参数。</li>
<li><strong>展望</strong>：参考 <a href="https://github.com/golang/go/issues/49085">Issue #49085</a>，尽管实现难度极大（涉及运行时字典传递或单态化膨胀），但核心团队将其列入规划，表明他们正在寻找突破口。一旦实现，像 Stream.Map[T, U](func(T) U) 这样流畅的链式调用将成为可能。</li>
</ul>
<h3>联合类型 (Union Types)</h3>
<ul>
<li><strong>关键词</strong>：union type</li>
<li><strong>解读</strong>：参考 <a href="https://github.com/golang/go/issues/19412">Issue #19412</a>，这不仅仅是泛型约束中的 A | B。真正的联合类型（类似 Rust 的 Enum 或 TypeScript 的 Union）可以让 Go 拥有更强大的模式匹配能力。配合可能的 match 语法，它将彻底改变 Go 的错误处理和状态机编写方式，使其更安全、更简洁。</li>
</ul>
<h3>Tensor (?) —— AI 时代的入场券</h3>
<ul>
<li><strong>关键词</strong>：maybe tensor (?)</li>
<li><strong>解读</strong>：这个带问号的项充满了想象力。它暗示 Go 团队可能正在严肃考虑为 <strong>AI/ML 工作负载</strong>提供原生的多维数组支持。如果 Go 能在语言层面原生支持高效的 Tensor 操作和自动微分，它将有资格挑战 Python 在 AI 基础设施领域的统治地位。当然这一切还只是猜测。</li>
</ul>
<hr />
<h2>工具链革命 —— 无痛 CGO</h2>
<h3>无 C 工具链的 CGO (CGO without C toolchain)</h3>
<ul>
<li><strong>关键词</strong>：cgo without C toolchain</li>
<li><strong>痛点</strong>：目前启用 CGO 就意味着必须安装 GCC/Clang，且失去了跨平台交叉编译的便利性（CGO_ENABLED=0 是多少 Gopher 的无奈之选）。</li>
<li><strong>解决方案</strong>：Go 团队的目标是实现<strong>“纯 Go 的 C 交互”</strong>。这可能通过两种路径实现：
<ul>
<li><strong>运行时加载</strong>：类似 purego，在运行时动态加载共享库并调用，无需编译期链接。</li>
<li><strong>内置微型链接器</strong>：Go 编译器直接解析 C 头文件并生成调用代码。</li>
<li>无论上述哪种方式，或是其他方式，一旦实现，<strong>“Write once, compile anywhere”</strong> 的承诺将在 CGO 场景下也得以兑现。</li>
</ul>
</li>
</ul>
<h3>Wasm 栈切换</h3>
<ul>
<li><strong>关键词</strong>：Wasm stack switching</li>
<li><strong>解读</strong>：这是为了更好地支持 <strong>Go 在浏览器中的异步模型</strong>。通过栈切换（Stack Switching），Go 可以更高效地挂起和恢复 Wasm 的执行，从而与 JavaScript 的 Promise 和 async/await 机制无缝互操作，显著减小 Wasm 产物的体积并提升性能。</li>
</ul>
<hr />
<h2>小结：性能与表达力的双重飞跃</h2>
<p>看完这份 2026 路线图，我们不禁感叹：Go 语言正在经历它的<strong>“成人礼”</strong>。</p>
<ul>
<li><strong>在性能上</strong>，它不再满足于“够用”，而是通过 SIMD、手动内存管理和亲和性调度，向 C/C++ 统治的“极致性能领域”发起冲击。</li>
<li><strong>在表达力上</strong>，它正在补齐泛型后的最后短板，通过泛型方法和联合类型，让代码更优雅、更安全。</li>
<li><strong>在体验上</strong>，它致力于抹平 CGO 和交叉编译的最后一道坎。</li>
</ul>
<p>这是一个野心勃勃的计划。如果这些特性在 2026 年真地能如期落地，Go 将不再仅仅是“云原生的语言”，它将成为一个<strong>全能、极致、且依旧简单</strong>的通用计算平台。</p>
<h2>参考资料</h2>
<ul>
<li>Go compiler and runtime meeting notes &#8211; https://github.com/golang/go/issues/43930#issuecomment-3576250284</li>
<li>Directly freeing user memory to reduce GC work &#8211; https://go.dev/design/74299-runtime-freegc</li>
<li>runtime, cmd/compile: add runtime.freegc and runtime.freegcTracked to reduce GC work &#8211; https://github.com/golang/go/issues/74299</li>
<li>715761: runtime: support runtime.freegc in size-specialized mallocs for noscan objects &#8211; https://go-review.googlesource.com/c/go/+/715761</li>
<li>simd: architecture-specific SIMD intrinsics under a GOEXPERIMENT &#8211; https://github.com/golang/go/issues/73787</li>
<li>proposal: sync: support for sharded values &#8211; https://github.com/golang/go/issues/18802</li>
<li>runtime: stronger affinity between G ↔ P ↔ M ↔ CPU?  &#8211; https://github.com/golang/go/issues/65694</li>
<li>https://github.com/golang/go/discussions/70257 &#8211; https://github.com/golang/go/discussions/70257</li>
<li>Region-based memory management &#8211; https://en.wikipedia.org/wiki/Region-based_memory_management</li>
<li>proposal: spec: add sum types / discriminated unions &#8211; https://github.com/golang/go/issues/19412</li>
<li>proposal: spec: allow type parameters in methods &#8211; https://github.com/golang/go/issues/49085</li>
</ul>
<hr />
<p>还在为“复制粘贴喂AI”而烦恼？我的新专栏 <strong>《<a href="http://gk.link/a/12EPd">AI原生开发工作流实战</a>》</strong> 将带你：</p>
<ul>
<li>告别低效，重塑开发范式</li>
<li>驾驭AI Agent(Claude Code)，实现工作流自动化</li>
<li>从“AI使用者”进化为规范驱动开发的“工作流指挥家”</li>
</ul>
<p>扫描下方二维码，开启你的AI原生开发之旅。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/ai-native-dev-workflow-qr.png" alt="" /></p>
<hr />
<p>你的Go技能，是否也卡在了“熟练”到“精通”的瓶颈期？</p>
<ul>
<li>想写出更地道、更健壮的Go代码，却总在细节上踩坑？</li>
<li>渴望提升软件设计能力，驾驭复杂Go项目却缺乏章法？</li>
<li>想打造生产级的Go服务，却在工程化实践中屡屡受挫？</li>
</ul>
<p>继《<a href="http://gk.link/a/10AVZ">Go语言第一课</a>》后，我的《<a href="http://gk.link/a/12yGY">Go语言进阶课</a>》终于在极客时间与大家见面了！</p>
<p>我的全新极客时间专栏 《<a href="http://gk.link/a/12yGY">Tony Bai·Go语言进阶课</a>》就是为这样的你量身打造！30+讲硬核内容，带你夯实语法认知，提升设计思维，锻造工程实践能力，更有实战项目串讲。</p>
<p>目标只有一个：助你完成从“Go熟练工”到“Go专家”的蜕变！ 现在就加入，让你的Go技能再上一个新台阶！</p>
<p><img src="https://tonybai.com/wp-content/uploads/course-card/iamtonybai-banner-2.gif" alt="" /></p>
<hr />
<p>商务合作方式：撰稿、出书、培训、在线课程、合伙创业、咨询、广告合作。如有需求，请扫描下方公众号二维码，与我私信联系。</p>
<p><img src="http://image.tonybai.com/img/tonybai/iamtonybai-wechat-qr.png" alt="" /></p>
<p style='text-align:left'>&copy; 2025, <a href='https://tonybai.com'>bigwhite</a>. 版权所有. </p>
]]></content:encoded>
			<wfw:commentRss>https://tonybai.com/2025/11/28/go-2026-roadmap-revealed/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>PGO 驱动的“动态逃逸分析”：w.Write(b) 中的切片逃逸终于有救了？</title>
		<link>https://tonybai.com/2025/11/13/proposal-dynamic-escapes/</link>
		<comments>https://tonybai.com/2025/11/13/proposal-dynamic-escapes/#comments</comments>
		<pubDate>Thu, 13 Nov 2025 00:28:37 +0000</pubDate>
		<dc:creator>bigwhite</dc:creator>
				<category><![CDATA[技术志]]></category>
		<category><![CDATA[#72036]]></category>
		<category><![CDATA[CallWrite]]></category>
		<category><![CDATA[ConditionalStackAllocation]]></category>
		<category><![CDATA[devirtualize]]></category>
		<category><![CDATA[DynamicEscapes]]></category>
		<category><![CDATA[GC压力]]></category>
		<category><![CDATA[Go]]></category>
		<category><![CDATA[Golang]]></category>
		<category><![CDATA[GoodWriter]]></category>
		<category><![CDATA[Go语言]]></category>
		<category><![CDATA[io.Writer]]></category>
		<category><![CDATA[IR]]></category>
		<category><![CDATA[LeakingWriter]]></category>
		<category><![CDATA[PGO]]></category>
		<category><![CDATA[ProfileGuidedOptimization]]></category>
		<category><![CDATA[runtime.free]]></category>
		<category><![CDATA[thepudds]]></category>
		<category><![CDATA[w.Write(b)]]></category>
		<category><![CDATA[WIP]]></category>
		<category><![CDATA[乐观的重写]]></category>
		<category><![CDATA[优化]]></category>
		<category><![CDATA[内存分配]]></category>
		<category><![CDATA[内存安全]]></category>
		<category><![CDATA[分配次数]]></category>
		<category><![CDATA[切片逃逸]]></category>
		<category><![CDATA[动态逃逸分析]]></category>
		<category><![CDATA[去虚拟化]]></category>
		<category><![CDATA[基准测试]]></category>
		<category><![CDATA[堆]]></category>
		<category><![CDATA[堆上分配]]></category>
		<category><![CDATA[堆内存分配]]></category>
		<category><![CDATA[快速路径]]></category>
		<category><![CDATA[性能优化]]></category>
		<category><![CDATA[性能提升]]></category>
		<category><![CDATA[性能顽疾]]></category>
		<category><![CDATA[慢速路径]]></category>
		<category><![CDATA[接口方法调用]]></category>
		<category><![CDATA[接口调用]]></category>
		<category><![CDATA[提案]]></category>
		<category><![CDATA[条件化分配]]></category>
		<category><![CDATA[栈上分配]]></category>
		<category><![CDATA[硬件潜能]]></category>
		<category><![CDATA[编译器]]></category>
		<category><![CDATA[编译器外科手术]]></category>
		<category><![CDATA[耗时]]></category>
		<category><![CDATA[运行时]]></category>
		<category><![CDATA[运行时profile数据]]></category>
		<category><![CDATA[逃逸]]></category>
		<category><![CDATA[闭包调用]]></category>
		<category><![CDATA[隐形开销]]></category>
		<category><![CDATA[零分配]]></category>
		<category><![CDATA[静态分析]]></category>
		<category><![CDATA[静态去虚拟化]]></category>

		<guid isPermaLink="false">https://tonybai.com/?p=5384</guid>
		<description><![CDATA[本文永久链接 &#8211; https://tonybai.com/2025/11/13/proposal-dynamic-escapes 大家好，我是Tony Bai。 io.Writer，这个在 Go 语言中无处不在的神圣接口，其背后却隐藏着一个困扰了性能敏感型开发者多年的“隐形成本”。当你将一个在函数内创建的字节切片 b 传递给 w.Write(b) 时，这个切片几乎总是会逃逸 (Escape) 到堆上，导致一次不必要的内存分配。 为什么？因为编译器不知道 w 的具体实现是什么，它必须做出最保守的假设。然而，一个由 Go 核心贡献者 thepudds 提交的新提案（#72036），正试图通过引入一种由 PGO (Profile-Guided Optimization) 驱动的“动态逃逸分析”新机制，来从根本上解决这个顽疾。 这项技术，真的能拯救 w.Write(b) 吗？它背后的原理又是什么？ 本文将深入剖析这场旨在消除接口调用隐形开销的编译器“外科手术”。 接口调用的性能“原罪”：保守的逃逸分析 让我们通过一个简单的基准测试，来直观地感受这个问题： package main import ( "io" "testing" ) // 一个“良好”的 Writer 实现，它不会保留传入的切片 type GoodWriter struct{} func (g *GoodWriter) Write(p []byte) (n int, err error) { [...]]]></description>
			<content:encoded><![CDATA[<p><img src="https://tonybai.com/wp-content/uploads/2025/proposal-dynamic-escapes-1.png" alt="" /></p>
<p><a href="https://tonybai.com/2025/11/13/proposal-dynamic-escapes">本文永久链接</a> &#8211; https://tonybai.com/2025/11/13/proposal-dynamic-escapes</p>
<p>大家好，我是Tony Bai。</p>
<p>io.Writer，这个在 Go 语言中无处不在的神圣接口，其背后却隐藏着一个困扰了性能敏感型开发者多年的“隐形成本”。当你将一个在函数内创建的字节切片 b 传递给 w.Write(b) 时，这个切片几乎总是会<strong>逃逸 (Escape)</strong> 到堆上，导致一次不必要的内存分配。</p>
<p>为什么？因为编译器不知道 w 的具体实现是什么，它必须做出最保守的假设。然而，一个由 Go 核心贡献者 thepudds 提交的新提案（<a href="https://github.com/golang/go/issues/72036">#72036</a>），正试图通过引入一种由 <strong>PGO (Profile-Guided Optimization) 驱动的“动态逃逸分析”</strong>新机制，来从根本上解决这个顽疾。</p>
<p>这项技术，真的能拯救 w.Write(b) 吗？它背后的原理又是什么？</p>
<p>本文将深入剖析这场旨在消除接口调用隐形开销的编译器“外科手术”。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/paid/go-micro-column-2025-pr.png" alt="" /></p>
<h2>接口调用的性能“原罪”：保守的逃逸分析</h2>
<p>让我们通过一个简单的基准测试，来直观地感受这个问题：</p>
<pre><code class="go">package main

import (
    "io"
    "testing"
)

// 一个“良好”的 Writer 实现，它不会保留传入的切片
type GoodWriter struct{}
func (g *GoodWriter) Write(p []byte) (n int, err error) {
    return len(p), nil // 只是假装写入，然后丢弃
}

// 核心函数
func CallWrite(w io.Writer, x byte) {
    // 这个切片的底层数组，目前会逃逸到堆上
    b := make([]byte, 0, 64)
    b = append(b, x)
    w.Write(b) // 问题就出在这行接口方法调用
}

func BenchmarkCallWrite(b *testing.B) {
    g := &amp;GoodWriter{}
    b.ReportAllocs()
    for i := 0; i &lt; b.N; i++ {
        CallWrite(g, 0)
    }
}
</code></pre>
<p>运行这个基准测试，你会得到如下结果(因机器和go版本不同而已)：</p>
<pre><code>BenchmarkCallWrite    31895619    47.36 ns/op    64 B/op    1 allocs/op
</code></pre>
<blockquote>
<p>注：在我的macOS 15.7.1以及Go 1.25.3下，只有关闭优化，才能看到那一次64字节的堆内存分配。</p>
</blockquote>
<p>尽管 GoodWriter 的实现极其简单，并没有对切片 b 做任何“出格”的事情，但每次调用 CallWrite 依然产生了一次 <strong>64 字节的堆分配</strong>。</p>
<p><strong>原因在于</strong>：当编译器分析 CallWrite 函数时，它只知道 w 是一个 io.Writer。它无法预知在运行时，w 的具体类型究竟是什么。万一传入的是一个“邪恶”的实现呢？</p>
<pre><code class="go">// 一个“邪恶”的 Writer，它会将切片泄露到一个全局变量中
var global []byte
type LeakingWriter struct{}
func (w *LeakingWriter) Write(p []byte) (n int, err error) {
    global = p // 切片被泄露了！
    return len(p), nil
}
</code></pre>
<p>为了保证内存安全，编译器必须采取最保守的策略：<strong>假设任何传递给接口方法调用的指针或切片，都可能会逃逸</strong>。因此，它只能将 b 的底层数组分配在堆上。这就是接口调用的性能“原罪”。</p>
<h2>新范式 —— PGO 如何赋能“条件化栈分配”</h2>
<p>提案 #72036 的核心思想，是让编译器变得更“聪明”，不再做出“一刀切”的最坏假设。它引入了一种被称为<strong>“动态逃逸” (Dynamic Escapes)</strong> 或<strong>“条件化栈分配” (Conditional Stack Allocation)</strong> 的新机制，并与 <strong>PGO</strong> 紧密结合。</p>
<p><strong>工作原理</strong>：</p>
<ol>
<li>
<p><strong>PGO 收集信息</strong>：当你开启 PGO 进行构建时，编译器会利用真实的运行时 profile 数据，分析出在 CallWrite 函数的调用点，w 这个接口变量<strong>最常见</strong>的具体类型是什么。假设 profile 显示，99% 的情况下，w 都是 *GoodWriter。</p>
</li>
<li>
<p><strong>编译器进行“去虚拟化(devirtualize)”重写</strong>：基于这份 profile 数据，编译器会在内部（IR 层面）对 w.Write(b) 的调用进行一次“乐观的”重写，其逻辑等价于：</p>
</li>
</ol>
<pre><code class="go">// 编译器在内部生成的伪代码
tmpw, ok := w.(*GoodWriter)
if ok {
    // 快速路径：我们“猜” w 是 *GoodWriter
    tmpw.Write(b) // 这是一个具体类型的方法调用！
} else {
    // 慢速路径：猜错了，走常规的接口调用
    w.Write(b)
}
</code></pre>
<ol>
<li>
<p><strong>逃逸分析的“升级”</strong>：新提案的关键，就是<strong>让逃逸分析能够理解这个 if-else 分支</strong>。</p>
<ul>
<li>在 if ok 的分支中，编译器现在可以明确地分析 (*GoodWriter).Write 的具体实现，并<strong>证明</strong>在这个分支中，切片 b <strong>不会逃逸</strong>。</li>
<li>在 else 分支中，编译器依然做出最坏的假设，认为 b <strong>会逃逸</strong>。</li>
</ul>
</li>
<li>
<p><strong>条件化分配</strong>：基于上述分析，编译器最终会生成一段神奇的代码，其逻辑等价于：</p>
</li>
</ol>
<pre><code class="go">// 编译器最终生成的伪代码
tmpw, ok := w.(*GoodWriter)
if ok {
    // 快速路径：在栈上分配 b！
    var b_stack [64]byte
    b := b_stack[:0]
    b = append(b, x)
    tmpw.Write(b)
} else {
    // 慢速路径：在堆上分配 b
    b := make([]byte, 0, 64)
    b = append(b, x)
    w.Write(b)
}
</code></pre>
<p>通过这种方式，对于那 99% 的常见情况，内存分配被成功地<strong>从堆转移到了栈</strong>，实现了零分配！</p>
<h2>实证 —— 10 倍性能提升背后的编译器魔法</h2>
<p>提案作者 thepudds 已经实现了一个原型，其基准测试结果令人振奋。在使用 PGO 开启这项优化后，我们最初的 benchmark 结果发生了翻天覆地的变化：</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/proposal-dynamic-escapes-2.png" alt="" /></p>
<p>是的，你没看错。通过让编译器变得更“智能”，一个看似无解的性能问题被很好解决，带来了<strong>数量级的性能提升</strong>。</p>
<h2>未来展望 —— 从“动态逃逸”到 runtime.free</h2>
<p>这个提案目前仍处于工作原型 (WIP) 阶段，但它为 Go 的未来性能优化，打开了一扇充满想象力的大门。</p>
<ul>
<li><strong>更广泛的应用</strong>：这种“条件化分配”的机制，未来可能扩展到更多场景，例如处理大小可变的切片、优化闭包调用等。</li>
<li><strong>运行时 free</strong>：提案作者还提到了一个更激进的探索——在 Go 运行时中引入一个内部的 runtime.free 函数。这可以让编译器在某些可以静态证明安全的情况下，实现对堆内存的<strong>手动释放和快速重用</strong>，从而进一步降低 GC 压力。目前runtime.free进展反倒更快，已经有多个cl被merge到tip版本中了，很大可能在Go 1.26版本以实验特性落地。</li>
<li><strong>静态去虚拟化(devirtualize)</strong>：这种基于类型信息进行优化的思路，未来甚至可能在没有 PGO 的情况下，通过更强的静态分析来实现。</li>
</ul>
<h2>小结</h2>
<p>NO.72036 提案是 Go 编译器和运行时近年来在性能优化领域最令人兴奋的探索之一。它不再满足于对具体代码模式的“小修小补”，而是试图从根本上，通过赋予逃逸分析“理解”控制流和运行时类型信息的能力，来解决一整类长期存在的性能顽疾。</p>
<p>虽然这项功能何时能进入正式版尚无定论，但它清晰地指明了 Go 团队的演进方向：<strong>在保持语言简洁性的同时，通过让编译器和工具链变得越来越“聪明”，来持续压榨硬件的每一分潜能。</strong> w.Write(b) 中的切片逃逸问题，看起来终于有救了。</p>
<hr />
<p>你的Go技能，是否也卡在了“熟练”到“精通”的瓶颈期？</p>
<ul>
<li>想写出更地道、更健壮的Go代码，却总在细节上踩坑？</li>
<li>渴望提升软件设计能力，驾驭复杂Go项目却缺乏章法？</li>
<li>想打造生产级的Go服务，却在工程化实践中屡屡受挫？</li>
</ul>
<p>继《<a href="http://gk.link/a/10AVZ">Go语言第一课</a>》后，我的《<a href="http://gk.link/a/12yGY">Go语言进阶课</a>》终于在极客时间与大家见面了！</p>
<p>我的全新极客时间专栏 《<a href="http://gk.link/a/12yGY">Tony Bai·Go语言进阶课</a>》就是为这样的你量身打造！30+讲硬核内容，带你夯实语法认知，提升设计思维，锻造工程实践能力，更有实战项目串讲。</p>
<p>目标只有一个：助你完成从“Go熟练工”到“Go专家”的蜕变！ 现在就加入，让你的Go技能再上一个新台阶！</p>
<p><img src="https://tonybai.com/wp-content/uploads/course-card/iamtonybai-banner-2.gif" alt="" /></p>
<hr />
<p>商务合作方式：撰稿、出书、培训、在线课程、合伙创业、咨询、广告合作。如有需求，请扫描下方公众号二维码，与我私信联系。</p>
<p><img src="http://image.tonybai.com/img/tonybai/iamtonybai-wechat-qr.png" alt="" /></p>
<p style='text-align:left'>&copy; 2025, <a href='https://tonybai.com'>bigwhite</a>. 版权所有. </p>
]]></content:encoded>
			<wfw:commentRss>https://tonybai.com/2025/11/13/proposal-dynamic-escapes/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>Go 的 16 年：一门为持久而生的编程语言</title>
		<link>https://tonybai.com/2025/11/12/16-years-of-go-a-programming-language-built-to-last/</link>
		<comments>https://tonybai.com/2025/11/12/16-years-of-go-a-programming-language-built-to-last/#comments</comments>
		<pubDate>Wed, 12 Nov 2025 00:25:02 +0000</pubDate>
		<dc:creator>bigwhite</dc:creator>
				<category><![CDATA[技术志]]></category>
		<category><![CDATA[AI基础设施]]></category>
		<category><![CDATA[AI服务]]></category>
		<category><![CDATA[API]]></category>
		<category><![CDATA[ArdanLabs]]></category>
		<category><![CDATA[CPU]]></category>
		<category><![CDATA[docker]]></category>
		<category><![CDATA[DWARF5]]></category>
		<category><![CDATA[encoding/json/v2]]></category>
		<category><![CDATA[FlightRecorder]]></category>
		<category><![CDATA[Go]]></category>
		<category><![CDATA[go1.0]]></category>
		<category><![CDATA[go1.18]]></category>
		<category><![CDATA[go1.25]]></category>
		<category><![CDATA[godoc]]></category>
		<category><![CDATA[gofmt]]></category>
		<category><![CDATA[Golang]]></category>
		<category><![CDATA[GOMAXPROCS]]></category>
		<category><![CDATA[gomod]]></category>
		<category><![CDATA[Google]]></category>
		<category><![CDATA[Gopher]]></category>
		<category><![CDATA[goroutine]]></category>
		<category><![CDATA[gotest]]></category>
		<category><![CDATA[goversion]]></category>
		<category><![CDATA[govet]]></category>
		<category><![CDATA[Go语言]]></category>
		<category><![CDATA[GreenTeaGC]]></category>
		<category><![CDATA[ignore指令]]></category>
		<category><![CDATA[KenThompson]]></category>
		<category><![CDATA[Kubernetes]]></category>
		<category><![CDATA[macOS12]]></category>
		<category><![CDATA[prometheus]]></category>
		<category><![CDATA[reddit]]></category>
		<category><![CDATA[RobertGriesemer]]></category>
		<category><![CDATA[RobPike]]></category>
		<category><![CDATA[terraform]]></category>
		<category><![CDATA[testing/synctest]]></category>
		<category><![CDATA[Web服务器]]></category>
		<category><![CDATA[WindowsARM]]></category>
		<category><![CDATA[云原生]]></category>
		<category><![CDATA[停顿时间]]></category>
		<category><![CDATA[克制]]></category>
		<category><![CDATA[分布式系统]]></category>
		<category><![CDATA[创新]]></category>
		<category><![CDATA[可观测性]]></category>
		<category><![CDATA[可靠]]></category>
		<category><![CDATA[后端系统]]></category>
		<category><![CDATA[向后兼容]]></category>
		<category><![CDATA[垃圾回收器]]></category>
		<category><![CDATA[安全性]]></category>
		<category><![CDATA[实用性]]></category>
		<category><![CDATA[容器感知]]></category>
		<category><![CDATA[工具链]]></category>
		<category><![CDATA[并发]]></category>
		<category><![CDATA[微服务]]></category>
		<category><![CDATA[快速]]></category>
		<category><![CDATA[快速编译]]></category>
		<category><![CDATA[性能]]></category>
		<category><![CDATA[性能分析]]></category>
		<category><![CDATA[持久]]></category>
		<category><![CDATA[数据流]]></category>
		<category><![CDATA[标准库]]></category>
		<category><![CDATA[核心类型]]></category>
		<category><![CDATA[泛型]]></category>
		<category><![CDATA[清晰]]></category>
		<category><![CDATA[稳定性]]></category>
		<category><![CDATA[空指针]]></category>
		<category><![CDATA[简洁]]></category>
		<category><![CDATA[简洁性]]></category>
		<category><![CDATA[类型推断]]></category>
		<category><![CDATA[编程语言]]></category>
		<category><![CDATA[编译器]]></category>
		<category><![CDATA[设计哲学]]></category>
		<category><![CDATA[调试信息]]></category>
		<category><![CDATA[边缘计算]]></category>
		<category><![CDATA[运行时]]></category>
		<category><![CDATA[追踪]]></category>
		<category><![CDATA[遥测]]></category>
		<category><![CDATA[里程碑]]></category>
		<category><![CDATA[链接器]]></category>
		<category><![CDATA[长远思考]]></category>
		<category><![CDATA[静态类型]]></category>
		<category><![CDATA[高性能]]></category>

		<guid isPermaLink="false">https://tonybai.com/?p=5380</guid>
		<description><![CDATA[本文永久链接 &#8211; https://tonybai.com/2025/11/12/16-years-of-go-a-programming-language-built-to-last 大家好，我是Tony Bai。 每年的十一月，对于全球的 Gopher 而言，都是一个值得纪念的特殊时刻。今年，我们迎来了 Go 语言公开发布的第 16 个年头。 在众多的庆祝文章中，来自 Go 社区的知名组织 Ardan Labs 发布的这篇《Go 的 16 年：一门为持久而生的编程语言》，以其深邃的洞察力和饱满的情感，深深地打动了我们。 这篇文章不仅仅是对 Go 历史里程碑的简单罗列，更是一次对 Go 设计哲学——克制、清晰与长远思考——的深刻致敬。文章精准地捕捉了 Go 从解决 Google 内部的工程困境，到成为现代云原生基石的宏大叙事。我们相信，无论对于已经与 Go 同行多年的资深开发者，还是刚刚踏上 Gopher 之旅的新人，这篇文章都能带来启发与共鸣。 为此，我特将其全文翻译为中文，希望能与中文 Go 社区的各位一同分享这份喜悦与思考。以下是正文： 每年的十一月，Go 社区都会为我们这个时代最具悄然变革力量的编程语言之一，庆祝又一个里程碑。 诞生于 Google 并于 2009 年向世界发布的 Go，旨在解决大规模软件构建、庞大代码库、分布式系统以及跨大洲团队协作的复杂性。十六年后的今天，Go 诞生之初秉持的原则——简洁、快速和可靠——依然指导着它的发展。 正如 Go 团队在去年的周年纪念博文中所写：“Go 是为 2007 年的软件工程问题而构建的，但它仍在解决 2024 年的挑战，以及那些尚未到来的挑战。” 起源故事 [...]]]></description>
			<content:encoded><![CDATA[<p><img src="https://tonybai.com/wp-content/uploads/2025/16-years-of-go-a-programming-language-built-to-last-1.png" alt="" /></p>
<p><a href="https://tonybai.com/2025/11/12/16-years-of-go-a-programming-language-built-to-last">本文永久链接</a> &#8211; https://tonybai.com/2025/11/12/16-years-of-go-a-programming-language-built-to-last</p>
<p>大家好，我是Tony Bai。</p>
<p>每年的十一月，对于全球的 Gopher 而言，都是一个值得纪念的特殊时刻。今年，我们迎来了 Go 语言公开发布的第 16 个年头。</p>
<p>在众多的庆祝文章中，来自 Go 社区的知名组织 Ardan Labs 发布的这篇《<a href="https://www.ardanlabs.com/news/2025/16-years-of-go-a-programming-language-built-to-last">Go 的 16 年：一门为持久而生的编程语言</a>》，以其深邃的洞察力和饱满的情感，深深地打动了我们。</p>
<p>这篇文章不仅仅是对 Go 历史里程碑的简单罗列，更是一次对 Go 设计哲学——克制、清晰与长远思考——的深刻致敬。文章精准地捕捉了 Go 从解决 Google 内部的工程困境，到成为现代云原生基石的宏大叙事。我们相信，无论对于已经与 Go 同行多年的资深开发者，还是刚刚踏上 Gopher 之旅的新人，这篇文章都能带来启发与共鸣。</p>
<p>为此，我特将其全文翻译为中文，希望能与中文 Go 社区的各位一同分享这份喜悦与思考。以下是正文：</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/paid/the-ultimate-guide-to-go-module-qr.png" alt="" /></p>
<hr />
<p>每年的十一月，Go 社区都会为我们这个时代最具悄然变革力量的编程语言之一，庆祝又一个里程碑。</p>
<p>诞生于 Google 并于 <strong>2009</strong> 年向世界发布的 Go，旨在解决大规模软件构建、庞大代码库、分布式系统以及跨大洲团队协作的复杂性。十六年后的今天，Go 诞生之初秉持的原则——<strong>简洁、快速和可靠</strong>——依然指导着它的发展。</p>
<p>正如 <a href="https://tonybai.com/2024/11/12/go-turns-15">Go 团队在去年的周年纪念博文</a>中所写：“Go 是为 2007 年的软件工程问题而构建的，但它仍在解决 2024 年的挑战，以及那些尚未到来的挑战。”</p>
<h2>起源故事</h2>
<p>这门语言源于 Google 三位工程师——<strong>Robert Griesemer, Rob Pike, 和 Ken Thompson</strong>——的挫败感，他们想要一门像 C 一样快、像 Python 一样高效、并且能满足 Google 基础设施规模化需求的语言。</p>
<p>他们并不想彻底革新编程，他们只是想让编程再次变得令人愉悦。</p>
<p>正如Rob Pike曾经说过的那样，“Go 是一次关于我们能去除什么的实验。”他们去除的过度复杂性、无休止的编译时间和混乱的依赖关系，反而成为了 Go 最大的优势。</p>
<h2>Go 编程语言为何能迅速走红</h2>
<p>Go 不仅仅是又一门新语言；它是对<strong>过度工程化的一次宣言</strong>。其设计目标使其脱颖而出：</p>
<ul>
<li><strong>快速编译</strong>：代码在数秒内完成构建，而非数分钟。</li>
<li><strong>简洁性</strong>：极简的特性集，强调清晰与可读性。</li>
<li><strong>并发</strong>：轻量级的 goroutine，使并发编程变得实用。</li>
<li><strong>静态类型 + 安全性</strong>：在不牺牲开发速度的前提下，保证类型安全。</li>
<li><strong>一流的工具链</strong>：go fmt、go test、go mod 及其他工具，塑造了 Go 的工匠精神文化。</li>
</ul>
<p>这些价值观深深地触动了那些厌倦了语言功能蔓延的工程师们，也触动了那些需要稳定、可维护系统的公司。</p>
<h2>现实世界中的 Go</h2>
<p>多年来，Go 已悄然成为现代Web的支柱。它驱动着 <strong>Docker、Kubernetes、Terraform 和 Prometheus</strong>——当今云原生生态系统的根基。</p>
<p>在 Google 内部，它在后端系统中每秒处理数十亿次请求。在 Google 之外，它已成为初创公司构建分布式系统和企业级工具的首选，这些场景都要求在没有摩擦的情况下获得高性能。</p>
<blockquote>
<p>“Go 诞生于 14 年前，至今它仍是唯一一门让并发感觉如此简单的语言。”</p>
</blockquote>
<p>这种观点体现了 Go 在开发者领域中的独特地位：它既足够古老，经受住了考验，又足够现代，能够不断演进发展。</p>
<h2>值得庆祝的里程碑</h2>
<p>Go 的时间线上，点缀着一些关键时刻，展示了这门语言是如何有意识地演进的：</p>
<ul>
<li><strong>2009年</strong>：Google 正式公开发布 Go语言。</li>
<li><strong>2012年</strong>：Go 1.0 发布，并作出了向后兼容的承诺。</li>
<li><strong>2015–2018年</strong>：Go 成为容器化工具和微服务的标准。</li>
<li><strong>2022年</strong>：泛型在 Go 1.18 中到来——一个期待已久的里程碑。</li>
<li><strong>2024年</strong>：Go 位列全球最常用的十大语言之一，并在 AI 服务和边缘计算领域的采用率迅速增长。</li>
</ul>
<p>正是这种稳定性，加上审慎的创新，让 Go 得以经久不衰。当其他语言追逐潮流时，Go 始终立足于实用性。</p>
<h2>是什么让 Go 与众不同</h2>
<p>与许多在每个新版本中不断膨胀的现代语言不同，Go 的演进一直很保守，而这种克制最终得到了回报。</p>
<p>Go 团队保持了一种罕见的、对向后兼容的承诺。十年前编写的代码，今天依然可以编译和运行。对于那些需要跨越数年甚至数十年维护生产系统的组织来说，这种信任是无价的。</p>
<p>Go 的简洁性也促进了团队协作。开发者可以快速上手代码库并投入工作。没有无休止的语法或模式争论，只有简洁、直接且高效的代码。</p>
<p>这种清晰性塑造了一个重视协作而非“炫技”的社区。</p>
<h2>社区的经验教训</h2>
<p>在一份以前的 <a href="https://www.reddit.com/r/golang/comments/17rx47o/go_was_announced_exactly_14_years_ago_happy/">Reddit 周年纪念帖子</a> 中，开发者们回顾了 Go 是如何改变他们职业生涯的：</p>
<blockquote>
<p>“Go 让我重新爱上了编程。”</p>
<p>“它不花哨，但它能搞定事情，这就是我爱它的地方。”</p>
</blockquote>
<p>这些故事体现了 Go 的不朽精神；与其说是炒作，不如说是把工作做好。</p>
<h2>下一章</h2>
<p>Go 的下一个十年，将不仅仅是关于 Web 服务器和 API。其生态系统正在扩展到<strong>AI 基础设施、数据流</strong>和<strong>边缘计算</strong>等领域，在这些地方，性能、并发和简洁性至关重要。</p>
<p>根据 Go 团队的 15 周年博文，当前的工作重点是：</p>
<ul>
<li>利用现代 CPU 架构，优化运行时性能。</li>
<li>改进生产系统中的遥测、可观测性和性能分析。</li>
<li>确保 Go 能够随着下一代硬件的发展而持续扩展。</li>
</ul>
<p>对于押注 Go 的开发者和组织来说，这意味着一件事：这门语言没有放慢脚步，它正在升级。</p>
<h2>Go的2025年：稳步求精，基础更牢固</h2>
<p>发布于 2025 年 8 月的 Go 1.25 版本，体现了这门语言标志性的演进方式——安静、审慎的改进，而非颠覆。虽然没有破坏性变更，但几项更新有意义地加固了 Go 的基础。通过移除旧的“core type”概念，语言规范得以简化，澄清了类型推断和泛型的工作方式。工具链变得更精简、更快速，工具现在按需构建，go.mod 中加入了新的ignore指令，同时 go vet, go doc, 和 go version 等命令也得到了增强。</p>
<p>在底层，运行时获得了容器感知能力，能够根据 CPU 限制自动调整 GOMAXPROCS，使 Go 在云和边缘环境中更加高效。一个新的实验性垃圾回收器（greenteagc）提供了明显更低的停顿时间，而“ Flight Recorder”追踪则引入了持续的、低开销的可观测性。编译器和链接器现在能生成 DWARF 5 调试信息，以获得更小的二进制文件和更快的构建速度，同时修复了一个微妙的空指针 bug，提升了运行时安全。</p>
<p>在标准库中，开发者现在可以通过 testing/synctest 更容易地测试并发代码，并可以试用更快、更灵活的 encoding/json/v2 包。平台支持也向前迈进——现在要求 macOS 12 或更新版本，而 32 位 Windows ARM 将在此版本后停止支持。</p>
<p>总而言之，Go 1.25 提醒了我们这门语言为何能经久不衰：它在不破坏信任的前提下演进，用稳定、有影响力的进步，取代了喧嚣的炒作。</p>
<p>（来源: <a href="https://go.dev/doc/go1.25?utm_source=chatgpt.com">go.dev/doc/go1.25</a>）</p>
<h2>为 Go 干杯</h2>
<p>在 Go 语言诞生 16 周年之际，我们不妨停下来，细细品味它所代表的意义。它不仅仅是一门编程语言，更是一种工程理念，其核心在于克制、清晰和长远思考。</p>
<p>在 Ardan Labs，我们亲眼见证了 Go 如何帮助团队构建可靠、可扩展的系统，从企业平台到初创原型，无所不包。它帮助工程师专注于真正重要的事情：解决实际问题，而不是与工具较劲。</p>
<p>祝愿 Go 语言再创辉煌一年。</p>
<p><strong>不追逐潮流的语言，才能超越潮流而长存。</strong></p>
<hr />
<p>你的Go技能，是否也卡在了“熟练”到“精通”的瓶颈期？</p>
<ul>
<li>想写出更地道、更健壮的Go代码，却总在细节上踩坑？</li>
<li>渴望提升软件设计能力，驾驭复杂Go项目却缺乏章法？</li>
<li>想打造生产级的Go服务，却在工程化实践中屡屡受挫？</li>
</ul>
<p>继《<a href="http://gk.link/a/10AVZ">Go语言第一课</a>》后，我的《<a href="http://gk.link/a/12yGY">Go语言进阶课</a>》终于在极客时间与大家见面了！</p>
<p>我的全新极客时间专栏 《<a href="http://gk.link/a/12yGY">Tony Bai·Go语言进阶课</a>》就是为这样的你量身打造！30+讲硬核内容，带你夯实语法认知，提升设计思维，锻造工程实践能力，更有实战项目串讲。</p>
<p>目标只有一个：助你完成从“Go熟练工”到“Go专家”的蜕变！ 现在就加入，让你的Go技能再上一个新台阶！</p>
<p><img src="https://tonybai.com/wp-content/uploads/course-card/iamtonybai-banner-2.gif" alt="" /></p>
<hr />
<p><strong>想系统学习Go，构建扎实的知识体系？</strong></p>
<p>我的新书《<a href="https://book.douban.com/subject/37499496/">Go语言第一课</a>》是你的首选。源自2.4万人好评的极客时间专栏，内容全面升级，同步至Go 1.24。首发期有专属五折优惠，不到40元即可入手，扫码即可拥有这本300页的Go语言入门宝典，即刻开启你的Go语言高效学习之旅！</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/go-primer-published-4.png" alt="" /></p>
<hr />
<p>商务合作方式：撰稿、出书、培训、在线课程、合伙创业、咨询、广告合作。如有需求，请扫描下方公众号二维码，与我私信联系。</p>
<p><img src="http://image.tonybai.com/img/tonybai/iamtonybai-wechat-qr.png" alt="" /></p>
<p style='text-align:left'>&copy; 2025, <a href='https://tonybai.com'>bigwhite</a>. 版权所有. </p>
]]></content:encoded>
			<wfw:commentRss>https://tonybai.com/2025/11/12/16-years-of-go-a-programming-language-built-to-last/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>Go 标准库将迎来 Zstandard：性能超越 Gzip，让你的应用更快、更省</title>
		<link>https://tonybai.com/2025/11/08/proposal-zstd/</link>
		<comments>https://tonybai.com/2025/11/08/proposal-zstd/#comments</comments>
		<pubDate>Fri, 07 Nov 2025 23:42:24 +0000</pubDate>
		<dc:creator>bigwhite</dc:creator>
				<category><![CDATA[技术志]]></category>
		<category><![CDATA[AddDict]]></category>
		<category><![CDATA[AI]]></category>
		<category><![CDATA[AI模型镜像]]></category>
		<category><![CDATA[ANS]]></category>
		<category><![CDATA[API]]></category>
		<category><![CDATA[API设计]]></category>
		<category><![CDATA[AsymmetricNumeralSystems]]></category>
		<category><![CDATA[BestCompression]]></category>
		<category><![CDATA[BestSpeed]]></category>
		<category><![CDATA[Brotli]]></category>
		<category><![CDATA[BurrowsWheeler]]></category>
		<category><![CDATA[close]]></category>
		<category><![CDATA[cloudflare]]></category>
		<category><![CDATA[CPU]]></category>
		<category><![CDATA[CPU占用]]></category>
		<category><![CDATA[DefaultCompression]]></category>
		<category><![CDATA[DEFLATE]]></category>
		<category><![CDATA[Dict]]></category>
		<category><![CDATA[Discord]]></category>
		<category><![CDATA[dsnet]]></category>
		<category><![CDATA[FaceBook]]></category>
		<category><![CDATA[Flush]]></category>
		<category><![CDATA[github]]></category>
		<category><![CDATA[Go]]></category>
		<category><![CDATA[Golang]]></category>
		<category><![CDATA[Golike]]></category>
		<category><![CDATA[GoTeam]]></category>
		<category><![CDATA[gzip]]></category>
		<category><![CDATA[Huffman]]></category>
		<category><![CDATA[issue]]></category>
		<category><![CDATA[jsonv2]]></category>
		<category><![CDATA[KlausPost]]></category>
		<category><![CDATA[LempelZivWelch]]></category>
		<category><![CDATA[lz4]]></category>
		<category><![CDATA[LZ77]]></category>
		<category><![CDATA[LZMA]]></category>
		<category><![CDATA[meta]]></category>
		<category><![CDATA[NoCompression]]></category>
		<category><![CDATA[ParseDict]]></category>
		<category><![CDATA[proposal]]></category>
		<category><![CDATA[reader]]></category>
		<category><![CDATA[Reset]]></category>
		<category><![CDATA[rfc]]></category>
		<category><![CDATA[RussCox]]></category>
		<category><![CDATA[SetLevel]]></category>
		<category><![CDATA[SetRawDict]]></category>
		<category><![CDATA[Setter模式]]></category>
		<category><![CDATA[snappy]]></category>
		<category><![CDATA[TonyBai]]></category>
		<category><![CDATA[websocket]]></category>
		<category><![CDATA[Websocket流量]]></category>
		<category><![CDATA[writer]]></category>
		<category><![CDATA[XZ]]></category>
		<category><![CDATA[Zstandard]]></category>
		<category><![CDATA[一致性]]></category>
		<category><![CDATA[优先级]]></category>
		<category><![CDATA[低延迟应用]]></category>
		<category><![CDATA[健壮性]]></category>
		<category><![CDATA[入门宝典]]></category>
		<category><![CDATA[内部会议]]></category>
		<category><![CDATA[升级]]></category>
		<category><![CDATA[协作]]></category>
		<category><![CDATA[协作精神]]></category>
		<category><![CDATA[压缩]]></category>
		<category><![CDATA[压缩比]]></category>
		<category><![CDATA[压缩算法]]></category>
		<category><![CDATA[反馈]]></category>
		<category><![CDATA[可审查性]]></category>
		<category><![CDATA[可维护性]]></category>
		<category><![CDATA[吞吐量]]></category>
		<category><![CDATA[天选之子]]></category>
		<category><![CDATA[字典功能]]></category>
		<category><![CDATA[字典预处理]]></category>
		<category><![CDATA[学习之旅]]></category>
		<category><![CDATA[学习成本]]></category>
		<category><![CDATA[安全优先]]></category>
		<category><![CDATA[安全可靠]]></category>
		<category><![CDATA[完整历程]]></category>
		<category><![CDATA[官方维护]]></category>
		<category><![CDATA[实战项目]]></category>
		<category><![CDATA[实时网关]]></category>
		<category><![CDATA[审查]]></category>
		<category><![CDATA[客户端带宽]]></category>
		<category><![CDATA[容器镜像]]></category>
		<category><![CDATA[工业界]]></category>
		<category><![CDATA[工程实践能力]]></category>
		<category><![CDATA[带宽]]></category>
		<category><![CDATA[带宽节约]]></category>
		<category><![CDATA[平衡]]></category>
		<category><![CDATA[应用]]></category>
		<category><![CDATA[开发者]]></category>
		<category><![CDATA[开销问题]]></category>
		<category><![CDATA[性能]]></category>
		<category><![CDATA[性能提升]]></category>
		<category><![CDATA[成功案例]]></category>
		<category><![CDATA[成本削减]]></category>
		<category><![CDATA[成熟]]></category>
		<category><![CDATA[技术优势]]></category>
		<category><![CDATA[技术选型]]></category>
		<category><![CDATA[拉取时间]]></category>
		<category><![CDATA[提案]]></category>
		<category><![CDATA[效率]]></category>
		<category><![CDATA[新书]]></category>
		<category><![CDATA[最佳实践]]></category>
		<category><![CDATA[标准库]]></category>
		<category><![CDATA[核心原则]]></category>
		<category><![CDATA[核心团队]]></category>
		<category><![CDATA[模式探索]]></category>
		<category><![CDATA[正式规范]]></category>
		<category><![CDATA[正确性]]></category>
		<category><![CDATA[流式Zstandard]]></category>
		<category><![CDATA[测试]]></category>
		<category><![CDATA[熵编码技术]]></category>
		<category><![CDATA[现代化]]></category>
		<category><![CDATA[生态演进]]></category>
		<category><![CDATA[知识体系]]></category>
		<category><![CDATA[社区]]></category>
		<category><![CDATA[社区英雄]]></category>
		<category><![CDATA[稳健]]></category>
		<category><![CDATA[简洁]]></category>
		<category><![CDATA[纯Go版本]]></category>
		<category><![CDATA[编译器]]></category>
		<category><![CDATA[翅膀]]></category>
		<category><![CDATA[解压]]></category>
		<category><![CDATA[设计思维]]></category>
		<category><![CDATA[语法认知]]></category>
		<category><![CDATA[语言]]></category>
		<category><![CDATA[语言生态]]></category>
		<category><![CDATA[贡献]]></category>
		<category><![CDATA[跨平台]]></category>
		<category><![CDATA[迁移成本]]></category>
		<category><![CDATA[运行时]]></category>
		<category><![CDATA[通用场景]]></category>
		<category><![CDATA[速度]]></category>
		<category><![CDATA[配置]]></category>
		<category><![CDATA[重大项目]]></category>
		<category><![CDATA[错误处理]]></category>
		<category><![CDATA[霍夫曼编码]]></category>
		<category><![CDATA[静态二进制文件]]></category>
		<category><![CDATA[静态字典]]></category>
		<category><![CDATA[领袖]]></category>

		<guid isPermaLink="false">https://tonybai.com/?p=5366</guid>
		<description><![CDATA[本文永久链接 &#8211; https://tonybai.com/2025/11/08/proposal-zstd 大家好，我是Tony Bai。 在 Go 的世界里，一项被社区翘首以盼的提案在沉寂一年后，终于迎来了决定性的进展。2024 年，将 Zstandard 压缩算法纳入标准库的提案（#62513）被正式 Accept，但在那之后便鲜有动静。直到最近的 Go 编译器与运行时会议纪要中透露，这项工作将由社区的明星开发者 Klaus Post 主导推进。 这意味着，在未来的 Go 版本中，开发者将能开箱即用地获得一个官方维护、安全可靠且性能卓越的压缩工具。这不仅是对 Go 生态的一次重要补强，更将直接为无数 Go 应用带来性能提升、带宽节约和成本削减，真正实现“更快、更省”的承诺。 同时，这个提案背后曲折的历程——从激烈的技术选型辩论，到精雕细琢的 API 设计，再到因核心团队资源紧张而搁置，最终由社区力量重新激活——本身就是一幅展现 Go 生态演进的生动图景。 在本文中，我们将探讨 Zstandard 脱颖而出的技术优势，剖析其在工业界的成功案例，并揭示 compress/zstd 标准库从提案、API 设计到最终由社区力量重启的完整历程。 Zstandard：为何是它，而非其他？ 在决定为标准库引入新的压缩算法时，Go 团队面临着众多选择。提案发起者 dsnet 在讨论中进行了一次精彩的“选美”，清晰地阐述了为何 Zstandard (Zstd) 能够脱颖而出： Zstandard (Zstd): 由 Facebook (现 Meta) 开发并开源，拥有极佳的压缩/解压速度和出色的压缩比。更重要的是，它有正式的 RFC 规范（RFC 8878），这对于标准库实现的“正确性”至关重要。 Brotli: 同样优秀，但在设计上更偏向 [...]]]></description>
			<content:encoded><![CDATA[<p><img src="https://tonybai.com/wp-content/uploads/2025/proposal-zstd-1.png" alt="" /></p>
<p><a href="https://tonybai.com/2025/11/08/proposal-zstd">本文永久链接</a> &#8211; https://tonybai.com/2025/11/08/proposal-zstd</p>
<p>大家好，我是Tony Bai。</p>
<p>在 Go 的世界里，一项被社区翘首以盼的提案在沉寂一年后，终于迎来了决定性的进展。2024 年，将 Zstandard 压缩算法纳入标准库的提案（<a href="https://github.com/golang/go/issues/62513">#62513</a>）被正式 <strong>Accept</strong>，但在那之后便鲜有动静。直到最近的 <a href="https://github.com/golang/go/issues/43930#issuecomment-3487773597">Go 编译器与运行时会议纪要</a>中透露，这项工作将由社区的明星开发者 Klaus Post 主导推进。</p>
<p>这意味着，在未来的 Go 版本中，开发者将能开箱即用地获得一个官方维护、安全可靠且性能卓越的压缩工具。这不仅是对 Go 生态的一次重要补强，更将直接为无数 Go 应用带来性能提升、带宽节约和成本削减，真正实现“更快、更省”的承诺。</p>
<p>同时，这个提案背后曲折的历程——从激烈的技术选型辩论，到精雕细琢的 API 设计，再到因核心团队资源紧张而搁置，最终由社区力量重新激活——本身就是一幅展现 Go 生态演进的生动图景。</p>
<p>在本文中，我们将探讨 Zstandard 脱颖而出的技术优势，剖析其在工业界的成功案例，并揭示 compress/zstd 标准库从提案、API 设计到最终由社区力量重启的完整历程。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/paid/the-ultimate-guide-to-go-module-qr.png" alt="" /></p>
<h2>Zstandard：为何是它，而非其他？</h2>
<p>在决定为标准库引入新的压缩算法时，Go 团队面临着众多选择。提案发起者 dsnet 在讨论中进行了一次精彩的“选美”，清晰地阐述了为何 Zstandard (Zstd) 能够脱颖而出：</p>
<ul>
<li><strong>Zstandard (Zstd):</strong> 由 Facebook (现 Meta) 开发并开源，拥有极佳的压缩/解压速度和出色的压缩比。更重要的是，它有正式的 RFC 规范（<a href="https://datatracker.ietf.org/doc/html/rfc8878">RFC 8878</a>），这对于标准库实现的“正确性”至关重要。</li>
<li><strong>Brotli:</strong> 同样优秀，但在设计上更偏向 Web 静态内容，且其庞大的静态字典（约 120KiB）与 Go 追求小体积静态二进制文件的哲学相悖。</li>
<li><strong>XZ (LZMA):</strong> 拥有极高的压缩比，但代价是极其缓慢的压缩和解压速度，不适合通用场景。且缺乏正式的、明确的规范。</li>
<li><strong>Snappy / LZ4:</strong> 追求极致的速度，但在压缩比上做出了巨大牺牲，应用场景相对小众。</li>
</ul>
<p>Zstd 巧妙地结合了 LZ77 算法和一种名为 ANS (Asymmetric Numeral Systems) 的现代熵编码技术，在性能、压缩比和资源消耗之间取得了近乎完美的平衡，使其成为替代 Gzip 的“天选之子”。</p>
<blockquote>
<p>注：截至Go 1.25.3版本，Go compress目录下提供了多种压缩算法的实现：bzip2实现了Burrows-Wheeler变换及霍夫曼编码；flate提供了DEFLATE算法核心，结合了LZ77和霍夫曼编码；gzip和zlib则分别将DEFLATE算法封装为gzip文件格式和zlib数据流格式；lzw实现了Lempel-Ziv-Welch算法。这些包共同为Go语言提供了多样化的数据压缩与解压缩能力。</p>
<p>注：Zstandard最新RFC规范为<a href="https://datatracker.ietf.org/doc/html/rfc9659">RFC 9659</a>。</p>
</blockquote>
<h2>工业界验证：Discord 与 Cloudflare 的性能飞跃</h2>
<p>理论上的优势必须经过实践的检验。Zstd 在工业界的应用早已硕果累累。</p>
<ul>
<li>
<p><a href="https://discord.com/blog/how-discord-reduced-websocket-traffic-by-40-percent">**Discord 的 40% 带宽削减</a>：** 通讯巨头 Discord 在将其实时网关的压缩算法从 zlib (Gzip) 迁移到<strong>流式 Zstandard</strong> 后，获得了惊人的收益。对于核心的 MESSAGE_CREATE 事件，压缩时间缩短了一半以上，负载体积也显著减小。这直接转化为更低的服务端 CPU 占用和客户端带宽节省，最终实现了 <strong>整体 Websocket 流量降低 40%</strong> 的壮举。</p>
</li>
<li>
<p><a href="https://blog.cloudflare.com/container-platform-preview">**Cloudflare 的容器镜像加速</a>：** 在其全球容器平台上，Cloudflare 需要快速分发巨大的 AI 模型镜像（常超过 15GB）。通过将镜像层压缩算法从 Gzip 更换为 Zstd，<strong>一个 30GB 镜像的拉取时间从 8 分钟骤降至 4 分钟</strong>，速度翻倍，极大地提升了全球调度的灵活性和响应速度。</p>
</li>
</ul>
<p>这些案例雄辩地证明，Zstd 是为现代高吞吐量、低延迟应用而生的。</p>
<h2>API 设计的艺术：一场关于简洁、安全与未来的辩论</h2>
<p>将新包引入标准库，API 的设计是重中之重。#62513 的讨论串完整记录了 compress/zstd API 从雏形到最终形态的演进过程。</p>
<h3>核心原则：安全与一致性</h3>
<p>提案伊始，就确立了两大基石：</p>
<ol>
<li><strong>安全优先：</strong> 标准库实现必须是<strong>纯 Go</strong>版本，不使用 unsafe 或汇编。dsnet 强调：“Go 社区调查一致显示，安全性比性能更重要。” 这意味着标准库版本追求的是可审查性、可维护性和跨平台的一致性，而非极致的性能。</li>
<li><strong>API 一致性：</strong> 新 API 应与 compress/gzip、compress/flate 等现有包保持风格统一，降低开发者的学习和迁移成本。</li>
</ol>
<h3>社区的声音：Klaus Post 的关键输入</h3>
<p>在讨论中，github.com/klauspost/compress 系列库的作者 <strong>Klaus Post</strong> 扮演了关键角色。他的库是 Go 社区公认的最高性能压缩实现，其丰富的实战经验为标准库的设计提供了宝贵视角。</p>
<p>Klaus 指出，他自己的库 API 相对复杂，是因为支持多线程、异步等高级特性。他赞同标准库应剥离这些复杂性，提供一个完全同步的、线程安全的 API。同时，他也对字典（Dictionary）功能的 API 设计提出了深刻见解，强调了字典预处理的开销问题，这直接影响了后续 API 的设计。</p>
<h3>最终定稿的 API</h3>
<p>经过多轮讨论，由 Russ Cox (rsc) 总结并最终被接受的 API 形态如下(并非最终版)：</p>
<pre><code class="go">package zstd

const (
    NoCompression      = 0
    BestSpeed          = 1
    BestCompression    = 9
    DefaultCompression = -1
)

type Dict struct { /* ... */ }
func ParseDict(enc []byte) (*Dict, error)
// ... 可能还包含 Marshal/Unmarshal 方法

type Reader struct { /* ... unexported fields ... */ }
func NewReader(r io.Reader) (*Reader, error)
func (z *Reader) Reset(r io.Reader) error
func (z *Reader) AddDict(*Dict)
func (z *Reader) SetRawDict([]byte)
func (z *Reader) Read(p []byte) (int, error)
func (z *Reader) Close() error

type Writer struct { /* ... unexported fields ... */ }
func NewWriter(w io.Writer) *Writer
func (z *Writer) Reset(w io.Writer)
func (z *Writer) SetLevel(int) error
func (z *Writer) AddDict(*Dict)
func (z *Writer) SetRawDict([]byte)
func (z *Writer) Write([]byte) (int, error)
func (z *Writer) Flush() error
func (z *Writer) Close() error
</code></pre>
<p>这个设计体现了 Go 标准库的哲学：</p>
<ul>
<li><strong>Setter 模式：</strong> 采用 SetLevel、AddDict 等方法进行配置，而不是更复杂的构造函数重载或函数式选项，兼顾了灵活性和简洁性。</li>
<li><strong>独立的 Dict 类型：</strong> 将字典抽象为 Dict 类型，通过 ParseDict 进行预处理。这解决了 Klaus 提出的“重复解析字典开销大”的问题，允许用户一次解析，多次复用。</li>
<li><strong>错误处理：</strong> 关键配置（如 SetLevel、ParseDict）返回 error，增强了 API 的健壮性。</li>
</ul>
<h2>漫长的等待与社区英雄的登场</h2>
<p>提案于 2024 年被接受，为何直到 2025 年底才真正启动？这背后反映了 Go 核心团队面临的现实挑战。Go 团队规模精简，核心成员的精力需要分配给语言、编译器、运行时等更高优先级的任务。提案发起者 dsnet 也深度参与了 json/v2 等重大项目，无暇分身。</p>
<p>在此期间，Klaus Post 主动请缨，表示愿意贡献一个精简版的、符合标准库要求的实现。然而，这个提议在当时并未得到明确的推进信号。</p>
<p>转机出现在 <a href="https://github.com/golang/go/issues/43930#issuecomment-3487773597">2025 年 11 月的 Go 团队内部会议</a>。纪要显示，团队终于有带宽来审查社区对 compress/flate 和 compress/zstd 的贡献。会议明确提到：“很高兴有社区审查。我们能去问问 k8s 的人吗？”（意指寻求更多社区的反馈和测试）。这标志着官方正式为 Klaus Post 的贡献打开了大门。随后Klaus Post也给出了自己的贡献时间表，大约在2026年Q1提交第一版实现给Go团队审查。</p>
<h2>小结：一次迟到但意义非凡的升级</h2>
<p>compress/zstd 的加入，对 Go 生态而言，是一次迟到但意义非凡的升级。它不仅仅是增加了一个功能包，更是一次：</p>
<ul>
<li><strong>技术的现代化：</strong> 用一个在性能和效率上全面超越 Gzip 的现代算法，武装 Go 的标准库。</li>
<li><strong>生态的成熟：</strong> 将社区经过千锤百炼的最佳实践，以安全、稳健的方式融入官方标准。</li>
<li><strong>模式的探索：</strong> 展示了在核心团队资源有限的情况下，如何通过与社区领袖的协作，共同推动语言生态向前发展。</li>
</ul>
<p>对于广大 Go 开发者来说，未来已来。不久之后（或许在 Go 1.27），我们将能以最简单、最 Go-like 的方式，为我们的应用插上 Zstandard 的翅膀，轻松实现性能提升与成本节约。这无疑是 Go 社区协作精神的又一次伟大胜利。</p>
<h2>参考资料</h2>
<ul>
<li>https://github.com/golang/go/issues/62513</li>
<li>https://blog.cloudflare.com/container-platform-preview </li>
<li>https://discord.com/blog/how-discord-reduced-websocket-traffic-by-40-percent </li>
<li>https://www.rfc-editor.org/rfc/rfc8878</li>
</ul>
<hr />
<p>你的Go技能，是否也卡在了“熟练”到“精通”的瓶颈期？</p>
<ul>
<li>想写出更地道、更健壮的Go代码，却总在细节上踩坑？</li>
<li>渴望提升软件设计能力，驾驭复杂Go项目却缺乏章法？</li>
<li>想打造生产级的Go服务，却在工程化实践中屡屡受挫？</li>
</ul>
<p>继《<a href="http://gk.link/a/10AVZ">Go语言第一课</a>》后，我的《<a href="http://gk.link/a/12yGY">Go语言进阶课</a>》终于在极客时间与大家见面了！</p>
<p>我的全新极客时间专栏 《<a href="http://gk.link/a/12yGY">Tony Bai·Go语言进阶课</a>》就是为这样的你量身打造！30+讲硬核内容，带你夯实语法认知，提升设计思维，锻造工程实践能力，更有实战项目串讲。</p>
<p>目标只有一个：助你完成从“Go熟练工”到“Go专家”的蜕变！ 现在就加入，让你的Go技能再上一个新台阶！</p>
<p><img src="https://tonybai.com/wp-content/uploads/course-card/iamtonybai-banner-2.gif" alt="" /></p>
<hr />
<p><strong>想系统学习Go，构建扎实的知识体系？</strong></p>
<p>我的新书《<a href="https://book.douban.com/subject/37499496/">Go语言第一课</a>》是你的首选。源自2.4万人好评的极客时间专栏，内容全面升级，同步至Go 1.24。首发期有专属五折优惠，不到40元即可入手，扫码即可拥有这本300页的Go语言入门宝典，即刻开启你的Go语言高效学习之旅！</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/go-primer-published-4.png" alt="" /></p>
<hr />
<p>商务合作方式：撰稿、出书、培训、在线课程、合伙创业、咨询、广告合作。如有需求，请扫描下方公众号二维码，与我私信联系。</p>
<p><img src="http://image.tonybai.com/img/tonybai/iamtonybai-wechat-qr.png" alt="" /></p>
<p style='text-align:left'>&copy; 2025, <a href='https://tonybai.com'>bigwhite</a>. 版权所有. </p>
]]></content:encoded>
			<wfw:commentRss>https://tonybai.com/2025/11/08/proposal-zstd/feed/</wfw:commentRss>
		<slash:comments>2</slash:comments>
		</item>
		<item>
		<title>Go 官方详解“Green Tea”垃圾回收器：从对象到页，一场应对现代硬件挑战的架构演进</title>
		<link>https://tonybai.com/2025/10/31/deep-into-go-green-tea-gc/</link>
		<comments>https://tonybai.com/2025/10/31/deep-into-go-green-tea-gc/#comments</comments>
		<pubDate>Thu, 30 Oct 2025 23:08:53 +0000</pubDate>
		<dc:creator>bigwhite</dc:creator>
				<category><![CDATA[技术志]]></category>
		<category><![CDATA[AVX512]]></category>
		<category><![CDATA[CherryMui]]></category>
		<category><![CDATA[CPUProfile]]></category>
		<category><![CDATA[CPU成本]]></category>
		<category><![CDATA[CPU时间]]></category>
		<category><![CDATA[CPU核心]]></category>
		<category><![CDATA[DavidChase]]></category>
		<category><![CDATA[GaliosField]]></category>
		<category><![CDATA[GitHubissue]]></category>
		<category><![CDATA[Go]]></category>
		<category><![CDATA[go1.25]]></category>
		<category><![CDATA[go1.26]]></category>
		<category><![CDATA[GOEXPERIMENT]]></category>
		<category><![CDATA[Golang]]></category>
		<category><![CDATA[Google]]></category>
		<category><![CDATA[GopherCon2025]]></category>
		<category><![CDATA[Go团队]]></category>
		<category><![CDATA[GraphFlood]]></category>
		<category><![CDATA[greentea]]></category>
		<category><![CDATA[GreenTeaGC]]></category>
		<category><![CDATA[Intel]]></category>
		<category><![CDATA[issue]]></category>
		<category><![CDATA[jsonv2]]></category>
		<category><![CDATA[KeithRandall]]></category>
		<category><![CDATA[map]]></category>
		<category><![CDATA[MarkSweepAlgorithm]]></category>
		<category><![CDATA[MichaelKnyszek]]></category>
		<category><![CDATA[MichaelPratt]]></category>
		<category><![CDATA[MicroarchitecturalDisaster]]></category>
		<category><![CDATA[nogreenteagc]]></category>
		<category><![CDATA[NUMA]]></category>
		<category><![CDATA[SIMD]]></category>
		<category><![CDATA[SIMDAccelerationPackage]]></category>
		<category><![CDATA[swisstable]]></category>
		<category><![CDATA[tip版本]]></category>
		<category><![CDATA[TracingGarbageCollection]]></category>
		<category><![CDATA[VGF2P8AFFINEQB]]></category>
		<category><![CDATA[WorkList]]></category>
		<category><![CDATA[x86硬件]]></category>
		<category><![CDATA[YvesVandriessche]]></category>
		<category><![CDATA[不可达]]></category>
		<category><![CDATA[主内存]]></category>
		<category><![CDATA[代码生成器]]></category>
		<category><![CDATA[仿射变换]]></category>
		<category><![CDATA[位]]></category>
		<category><![CDATA[位运算]]></category>
		<category><![CDATA[元数据]]></category>
		<category><![CDATA[先进先出]]></category>
		<category><![CDATA[全局变量]]></category>
		<category><![CDATA[内存]]></category>
		<category><![CDATA[内存带宽]]></category>
		<category><![CDATA[切片]]></category>
		<category><![CDATA[原型]]></category>
		<category><![CDATA[反馈]]></category>
		<category><![CDATA[后进先出]]></category>
		<category><![CDATA[向量加速]]></category>
		<category><![CDATA[向量增强]]></category>
		<category><![CDATA[向量指令]]></category>
		<category><![CDATA[向量硬件]]></category>
		<category><![CDATA[图]]></category>
		<category><![CDATA[垃圾回收]]></category>
		<category><![CDATA[垃圾回收器]]></category>
		<category><![CDATA[垃圾回收成本]]></category>
		<category><![CDATA[基准测试]]></category>
		<category><![CDATA[堆]]></category>
		<category><![CDATA[堆内存]]></category>
		<category><![CDATA[堆结构]]></category>
		<category><![CDATA[实验性]]></category>
		<category><![CDATA[寄存器]]></category>
		<category><![CDATA[对象]]></category>
		<category><![CDATA[局部变量]]></category>
		<category><![CDATA[工作负载]]></category>
		<category><![CDATA[已扫描位]]></category>
		<category><![CDATA[已见位]]></category>
		<category><![CDATA[已访问]]></category>
		<category><![CDATA[广度优先]]></category>
		<category><![CDATA[性能]]></category>
		<category><![CDATA[扫描]]></category>
		<category><![CDATA[技术原理]]></category>
		<category><![CDATA[指针]]></category>
		<category><![CDATA[架构演进]]></category>
		<category><![CDATA[标记]]></category>
		<category><![CDATA[标记阶段]]></category>
		<category><![CDATA[根]]></category>
		<category><![CDATA[汇编代码]]></category>
		<category><![CDATA[深度优先]]></category>
		<category><![CDATA[清除]]></category>
		<category><![CDATA[清除阶段]]></category>
		<category><![CDATA[现代硬件]]></category>
		<category><![CDATA[生产环境]]></category>
		<category><![CDATA[空闲槽位]]></category>
		<category><![CDATA[绿茶算法]]></category>
		<category><![CDATA[缓存]]></category>
		<category><![CDATA[编译器]]></category>
		<category><![CDATA[节点]]></category>
		<category><![CDATA[边]]></category>
		<category><![CDATA[运行时]]></category>
		<category><![CDATA[运行频率]]></category>
		<category><![CDATA[页]]></category>
		<category><![CDATA[默认垃圾回收器]]></category>

		<guid isPermaLink="false">https://tonybai.com/?p=5335</guid>
		<description><![CDATA[本文永久链接 &#8211; https://tonybai.com/2025/10/31/deep-into-go-green-tea-gc 大家好，我是Tony Bai。 关注 Go 语言演进的 Gopher 们可能已经注意到，Go 团队更换技术负责人以来，对运行时 (runtime) 和编译器 (compiler) 核心组件的打磨正日益成为团队的工作重心。从备受期待的“绿茶”GC (Green Tea GC)，到 标准库simd 加速包的探索，再到 基于swisstable的 map 的实现，以及 json/v2 的设计实现，一系列动作都预示着 Go 正在其性能核心地带进行着深刻的自我革新。 而就在最近，Go 运行时和编译器团队的一项决议，更是将这一趋势推向了高潮：他们计划在 Go 1.26 版本中，将实验性的“绿茶”GC 作为默认的垃圾回收器正式落地。 为了帮助大家深入理解这一重大变更背后的技术原理与深层思考，我翻译了 Go 官方博客10月29日的最新文章《The Green Tea Garbage Collector》。该文是基于 Go 团队核心成员 Michael Knyszek 在 GopherCon 2025 大会上的演讲整理而成。在这篇极具技术深度的原理文章中，没有人能比官方团队的讲解更为专业和权威。因此，为了最大程度地保留其“原汁原味”，我选择以全文翻译的形式，将其最真实、最精确的面貌呈现给大家。 以下是译文全文，供大家参考。 Go 1.25 包含一个名为“绿茶”（Green Tea）的全新实验性垃圾回收器，在构建时通过设置 GOEXPERIMENT=greenteagc 即可启用。使用该垃圾回收器后，许多工作负载在垃圾回收上花费的时间减少了约 [...]]]></description>
			<content:encoded><![CDATA[<p><img src="https://tonybai.com/wp-content/uploads/2025/deep-into-go-green-tea-gc-1.png" alt="" /></p>
<p><a href="https://tonybai.com/2025/10/31/deep-into-go-green-tea-gc">本文永久链接</a> &#8211; https://tonybai.com/2025/10/31/deep-into-go-green-tea-gc</p>
<p>大家好，我是Tony Bai。</p>
<p>关注 Go 语言演进的 Gopher 们可能已经注意到，<a href="https://tonybai.com/2024/10/10/pass-torch-to-go-new-leadership-team/">Go 团队更换技术负责人</a>以来，对运行时 (runtime) 和编译器 (compiler) 核心组件的打磨正日益成为团队的工作重心。从备受期待的<a href="https://tonybai.com/2025/05/03/go-green-tea-garbage-collector/">“绿茶”GC (Green Tea GC)</a>，到 <a href="https://tonybai.com/2025/08/22/go-simd-package-preview">标准库simd 加速包的探索</a>，再到 <a href="https://tonybai.com/2024/11/14/go-map-use-swiss-table/">基于swisstable的 map 的实现</a>，以及 <a href="https://tonybai.com/2025/08/09/true-streaming-support-in-jsonv2">json/v2</a> 的设计实现，一系列动作都预示着 Go 正在其性能核心地带进行着深刻的自我革新。</p>
<p>而就在最近，Go 运行时和编译器团队的一项决议，更是将这一趋势推向了高潮：<strong>他们计划在 Go 1.26 版本中，<a href="https://mp.weixin.qq.com/s/pjrnZQym724T5EGuL0a2UQ">将实验性的“绿茶”GC 作为默认的垃圾回收器正式落地</a>。</strong></p>
<p>为了帮助大家深入理解这一重大变更背后的技术原理与深层思考，我翻译了 <a href="https://tonybai.com/wp-content/uploads/2025">Go 官方博客10月29日的最新文章《The Green Tea Garbage Collector》</a>。该文是基于 Go 团队核心成员 Michael Knyszek 在 GopherCon 2025 大会上的演讲整理而成。在这篇极具技术深度的原理文章中，没有人能比官方团队的讲解更为专业和权威。因此，为了最大程度地保留其“原汁原味”，我选择以全文翻译的形式，将其最真实、最精确的面貌呈现给大家。</p>
<p>以下是译文全文，供大家参考。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/paid/the-ultimate-guide-to-go-module-qr.png" alt="" /></p>
<hr />
<p><a href="https://tonybai.com/2025/08/15/some-changes-in-go-1-25">Go 1.25</a> 包含一个名为“绿茶”（Green Tea）的全新实验性垃圾回收器，在构建时通过设置 GOEXPERIMENT=greenteagc 即可启用。使用该垃圾回收器后，许多工作负载在垃圾回收上花费的时间减少了约 10%，而有些工作负载的降幅甚至高达 40%！</p>
<p>它已为生产环境准备就绪，并在 Google 内部投入使用，因此我们鼓励你进行尝试。我们知道某些工作负载的收益不大，<a href="https://www.dolthub.com/blog/2025-09-26-greentea-gc-with-dolt/">甚至完全没有</a>，所以你的反馈对于我们向前推进至关重要。根据我们目前掌握的数据，我们计划<a href="https://mp.weixin.qq.com/s/pjrnZQym724T5EGuL0a2UQ">在 Go 1.26 中将其设为默认GC</a>。</p>
<p>如需报告任何问题，请<a href="https://go.dev/issue/new">提交一个新 issue</a>。</p>
<p>如需分享任何成功经验，请回复至<a href="https://go.dev/issue/73581">现有的 Green Tea issue</a>。</p>
<p>下文是基于 Michael Knyszek 在 GopherCon 2025 上的演讲整理的博文。一旦演讲视频上线，我们将会更新此博文并附上链接。</p>
<h2>追踪垃圾回收过程</h2>
<p>在讨论“绿茶”之前，让我们先就垃圾收集问题达成共识。</p>
<h3>对象和指针</h3>
<p>垃圾回收的目的是自动回收并重用程序不再使用的内存。</p>
<p>为此，Go 垃圾回收器关注的是对象(Object)和指针(Pointer)。</p>
<p>在 Go 运行时的上下文中，对象是Go值(Value)，其底层内存分配自堆。当 Go 编译器无法找到其他方式为某个值分配内存时，就会创建堆对象。例如，以下代码片段会分配一个堆对象：一个指针切片的底层存储空间。</p>
<pre><code>var x = make([]*int, 10) // 全局变量
</code></pre>
<p>Go 编译器只能在堆上分配切片后备存储，因为它很难（甚至可能不可能）知道 x 将引用该对象多长时间。</p>
<p>指针只是一些数字，用于指示 Go 值在内存中的位置，Go 程序通过它们来引用对象。例如，要获取上一个代码片段中分配的对象的起始指针，我们可以这样写：</p>
<pre><code>&amp;x[0] // 0xc000104000
</code></pre>
<h3>标记-清除算法</h3>
<p>Go 的垃圾回收器遵循一种广义上称为“追踪式垃圾回收”的策略，这意味着垃圾回收器会跟随或追踪程序中的指针，以识别程序仍在使用的对象。</p>
<p>更具体地说，Go 垃圾回收器实现了标记-清除(mark-sweep)算法。这比听起来要简单得多。 可以把对象和指针想象成计算机科学意义上的图：<strong>对象是节点，指针是边</strong>。</p>
<p>标记-清除算法就在这个图上运行的，顾名思义，它分两个阶段进行。</p>
<p>在第一阶段，即标记阶段，它从一组明确定义的、称为“根(root)”的源边开始遍历对象图。可以将其理解为<strong>全局变量</strong>和<strong>局部变量</strong>。然后，它将沿途找到的所有东西标记为<strong>已访问(visited)</strong>，以避免循环。这类似于典型的图遍历算法，如深度优先或广度优先搜索。</p>
<p>接下来是清除阶段。在我们的图遍历中未被访问到的任何对象，都是程序<strong>未使用</strong>或<strong>不可达(unreachable)</strong>的。我们称这种状态为<strong>不可达</strong>，因为通过语言的语义，正常的安全 Go 代码已无法再访问那块内存。为完成清除阶段，算法只需遍历所有未访问的节点，并将其内存标记为空闲，以便内存分配器可以重用它们。</p>
<h3>就是这样？</h3>
<p>你可能觉得我在这里把事情想得有点过于简单了。垃圾回收器经常被比作魔法和黑盒子 。你的说法也对了一部分，实际情况要复杂得多。</p>
<p>例如，实际上，这个算法会与你的常规 Go 代码并行执行。遍历一个不断变化的图会带来挑战。我们还对这个算法进行了并行化，这一点稍后会再次提及。</p>
<p>但请相信我，这些细节大多与核心算法无关。核心算法实际上只是一个简单的图泛洪(graph flood)操作。</p>
<h3>图泛洪示例</h3>
<p>我们来看一个例子。请浏览下面的幻灯片图片，跟随步骤操作。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-007.png" alt="" /></p>
<p>这里我们有一个包含一些全局变量和 Go 堆的图示。让我们一步步来分析。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-008.png" alt="" /></p>
<p>左边是我们的根。它们是全局变量 x 和 y。这将是我们图遍历的起点。根据左下角的图例，它们被标记为蓝色，表示它们当前在我们的工作列表上。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-009.png" alt="" /></p>
<p>右边是我们的堆。目前，堆中的所有东西都是灰色的，因为我们还没有访问过任何部分。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-010.png" alt="" /></p>
<p>每个矩形中代表一个对象。每个对象都标有其类型。这个特殊的对象是 T 类型的对象，其类型定义在左上角。它有一个指向子节点数组的指针和一些值。我们可以推断这是一种递归的树形数据结构。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-011.png" alt="" /></p>
<p>除了 T 类型的对象，你还会注意到我们有包含 *T 的数组对象。这些数组对象由 T 类型对象的 “children” 字段指向。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-012.png" alt="" /></p>
<p>矩形内的每个方块代表 8 字节的内存。带有点的方块是一个指针。如果它有箭头，那么它是一个指向某个其他对象的非空指针。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-013.png" alt="" /></p>
<p>如果它没有对应的箭头，那么它就是一个空指针。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-014.png" alt="" /></p>
<p>接下来，这些虚线矩形代表空闲空间，我称之为空闲“槽位(slot)”。我们可以在那里放置一个对象，但目前还没有。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-015.png" alt="" /></p>
<p>你还会注意到对象被这些带标签的、虚线圆角矩形组合在一起。每一个都代表一个页(page)：一块连续的内存块。这些页被标记为 A、B、C 和 D，我将以此来称呼它们。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-015.png" alt="" /></p>
<p>在这个图中，每个对象都被分配到某个页面中。就像实际实现一样，这里的每个页面只包含特定大小的对象。这正是 Go 堆的组织方式。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-016.png" alt="" /></p>
<p>页也是我们组织每个对象元数据的方式。这里你可以看到七个框，每个对应页 A 中的七个对象槽位之一。</p>
<p>每个框代表一位(bit)信息：我们之前是否见过这个对象。实际上，Go运行时就是通过这种方式来管理对象是否已被访问过的，这一点稍后会很重要。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-017.png" alt="" /></p>
<p>细节讲了很多，感谢你跟读。这些稍后都会派上用场。现在，让我们看看图泛洪如何应用于这幅图。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-018.png" alt="" /></p>
<p>我们首先从工作列表中取出一个根。我们将其标记为红色，表示它现在是活跃的。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-019.png" alt="" /></p>
<p>沿着根指针，我们找到了一个 T 类型的对象，并将其添加到我们的工作列表。根据图例，我们将该对象绘制成蓝色，以表明它已在工作列表中。请注意，我们同时在右上角的元数据中设置了与此对象对应的“已见”位。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-020.png" alt="" /></p>
<p>下一个根也同样处理。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-021.png" alt="" /></p>
<p>现在我们处理完了所有的根，工作列表上还剩下两个对象。让我们从工作列表中取出一个对象。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-022.png" alt="" /></p>
<p>我们现在要做的是遍历该对象的指针，以找到更多的对象。顺便说一下，我们称遍历一个对象的指针为“扫描”该对象。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-023.png" alt="" /></p>
<p>我们找到了这个有效的数组对象…</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-024.png" alt="" /></p>
<p>… 并将其添加到我们的工作列表中。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-025.png" alt="" /></p>
<p>从这里开始，我们递归地进行。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-026.png" alt="" /></p>
<p>我们遍历数组的指针。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-027.png" alt="" /></p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-028.png" alt="" /></p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-029.png" alt="" /></p>
<p>找到更多对象…</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-030.png" alt="" /></p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-031.png" alt="" /></p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-032.png" alt="" /></p>
<p>然后我们遍历数组对象引用的那些对象！</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-033.png" alt="" /></p>
<p>请注意，我们仍然需要遍历所有指针，即使它们是 nil。我们事先并不知道它们是否为空。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-034.png" alt="" /></p>
<p>这个分支下还有一个对象…</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-035.png" alt="" /></p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-036.png" alt="" /></p>
<p>现在我们到达了另一个分支，从我们早先从某个根找到的页 A 中的那个对象开始。</p>
<p>你可能注意到了我们工作列表的“后进先出”规则，这表明我们的工作列表是一个栈，因此我们的图遍历近似于深度优先。这是有意为之的，并反映了 Go 运行时中实际的图遍历算法。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-037.png" alt="" /></p>
<p>让我们继续…</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-038.png" alt="" /></p>
<p>接下来我们找到了另一个数组对象…</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-039.png" alt="" /></p>
<p>并遍历它…</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-040.png" alt="" /></p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-041.png" alt="" /></p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-042.png" alt="" /></p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-043.png" alt="" /></p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-044.png" alt="" /></p>
<p>我们的工作列表上只剩最后一个对象了…</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-045.png" alt="" /></p>
<p>让我们扫描它…</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-046.png" alt="" /></p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-047.png" alt="" /></p>
<p>标记阶段完成了！我们没有任何正在处理的工作，工作列表也空了。所有用黑色绘制的对象都是可达的，所有用灰色绘制的对象都是不可达的。让我们一次性清除所有不可达的对象。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/marksweep-048.png" alt="" /></p>
<p>我们已将那些对象转换为空闲槽位，准备好容纳新的对象。</p>
<h2>问题所在</h2>
<p>经过上面一番摸索，我认为我们已经掌握了 Go 垃圾回收器的实际工作原理。目前看来，这个过程运行良好，那么问题出在哪里呢？</p>
<p>事实证明，在某些程序中，执行这个特定算法会花费大量时间，而且几乎会给所有 Go 程序带来显著的开销。Go 程序将 20% 甚至更多的 CPU 时间用于垃圾回收的情况并不少见。</p>
<p>让我们来分析一下这些时间都花在了哪里。</p>
<h3>垃圾回收成本</h3>
<p>在宏观层面上，垃圾回收器的成本由两部分组成。一是运行频率，二是每次运行所做的工作量。将这两者相乘，就得到了垃圾回收的总成本。</p>
<pre><code>Total GC cost = Number of GC cycles × Average cost per GC cycle

即 总 GC 成本 = GC 周期数 × 每个 GC 周期的平均成本
</code></pre>
<p>多年来，我们一直在研究这个等式中的这两个术语。要了解更多关于垃圾回收器运行频率的信息，请参阅 <a href="https://www.youtube.com/watch?v=07wduWyWx8M">Michael 在 2022 年 GopherCon EU 大会上的关于内存限制的演讲</a>。 <a href="https://go.dev/doc/gc-guide">Go 垃圾回收器的指南</a>也对此主题进行了很多阐述，如果你想深入了解，值得一看。</p>
<p>但现在，我们只关注第二部分，即每个周期的成本。</p>
<p>多年来，我们不断研究 CPU Profile分析结果，试图提高性能，从中我们了解到 Go 的垃圾回收器有两大特点。</p>
<p>第一，大约 90% 的垃圾回收器成本都花在了标记上，只有大约 10% 是在清除。事实证明，清除比标记更容易优化，多年来 Go 已经拥有了一个非常高效的清除器。</p>
<p>第二，在那段用于标记的时间里，有相当大一部分(通常至少有 35%)，都浪费在了访问堆内存上。这本身已经够糟糕了，更糟糕的是，它完全阻碍了现代 CPU 真正高速运行的关键机制。</p>
<h3>“微架构灾难”</h3>
<p>在这种情况下，“堵塞工作机制(gump up the works)”意味着什么？现代 CPU 的具体构造相当复杂，所以我们用一个类比来说明。</p>
<p>想象 CPU 在一条路上行驶，这条路就是你的程序。CPU 想要加速到很高的速度，为此它需要能看清前方的路，并且道路必须畅通。但图遍历算法对 CPU 来说，就像在城市街道里开车。CPU 看不到拐角后的情况，也无法预测接下来会发生什么。为了前进，它必须不断地减速、转弯、在红绿灯前停下、避开行人。你的引擎有多快几乎无关紧要，因为你根本没有机会真正跑起来。</p>
<p>让我们通过再次审视我们的例子来使这一点更具体。我在这里的堆上叠加了我们所走的路径。每个从左到右的箭头代表我们做的一段扫描工作，虚线箭头则显示了我们在不同扫描工作之间是如何跳转的。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/graphflood-path.png" alt="" /></p>
<p>上图展示了我们的图泛洪示例中，垃圾回收器在堆中执行的路径。</p>
<p>请注意，我们正在内存中到处跳转，在每个地方只做一点点工作。特别是，我们频繁地在页之间，以及页的不同部分之间跳转。</p>
<p>现代 CPU 做了大量的缓存。访问主内存可能比访问缓存中的内存慢上 100 倍。CPU 缓存中填充的是最近访问过的内存，以及与最近访问过的内存相邻的内存。但是，并不能保证两个相互指向的对象在内存中也彼此靠近。图泛洪算法并没有考虑到这一点。</p>
<p>补充一点：如果我们只是在等待从主内存中获取数据，情况可能还没那么糟。CPU 会异步地发出内存请求，所以即使是慢的请求也可以重叠，只要 CPU 能看得足够远。但在图遍历中，每一小段工作都是不可预测的，并且高度依赖于上一段工作，所以 CPU 被迫几乎在每一次独立的内存获取后都进行等待。</p>
<p>不幸的是，对我们来说，这个问题只会越来越严重。业界有句格言：“等两年，你的代码会变得更快。”</p>
<p>但 Go，作为一个依赖于标记-清除算法的垃圾回收语言，却面临着相反的风险。“等两年，你的代码会变得更慢。” 现代 CPU 硬件的趋势正在给垃圾回收器的性能带来新的挑战：</p>
<ul>
<li>
<p><strong>非一致性内存访问 (Non-uniform memory access)。</strong> 首先，内存现在往往与 CPU 核心的子集相关联。其他 CPU 核心访问该内存的速度比前者慢。换句话说，主内存访问的成本<a href="https://jprahman.substack.com/p/sapphire-rapids-core-to-core-latency">取决于哪个 CPU 核心正在访问它</a> 。这种成本是不一致的，因此我们称之为非一致内存访问，简称 NUMA。</p>
</li>
<li>
<p><strong>内存带宽减少 (Reduced memory bandwidth)。</strong> 每个 CPU 的可用内存带宽随着时间推移呈下降趋势。这意味着虽然我们拥有更多的 CPU 核心，但每个核心能够提交的数据量相对较少。 对主内存的请求导致未缓存的请求等待时间比以前更长。</p>
</li>
<li>
<p><strong>越来越多的 CPU 核心 (Ever more CPU cores)。</strong> 上面，我们看的是一个顺序的标记算法，但真正的垃圾回收器是并行执行此算法的。这在核心数量有限的情况下扩展得很好，但即使经过精心设计，用于扫描的共享对象队列也会成为一个瓶颈。</p>
</li>
<li>
<p><strong>现代硬件特性 (Modern hardware features)。</strong> 新硬件拥有像向量指令这样的酷炫功能，让我们能一次性操作大量数据。虽然这有可能大幅提升速度，但目前还不清楚如何才能实现这一点。因为标记工作包含很多不规则且通常是小块的工作。</p>
</li>
</ul>
<h2>绿茶(Green Tea)</h2>
<p>最后，我们来看看绿茶算法，这是我们对标记扫描算法的一个新的尝试。绿茶算法的核心思想非常简单：</p>
<p><strong>操作页面，而不是对象。</strong></p>
<p>听起来很简单，对吧？然而，为了弄清楚如何安排对象图遍历的顺序以及我们需要跟踪哪些内容才能使其在实践中有效运作，我们做了大量的工作。</p>
<p>更具体地说，这意味着：</p>
<ul>
<li>我们不再扫描对象，而是扫描整个页。</li>
<li>我们不再在工作列表上跟踪对象，而是跟踪整个页。</li>
<li>我们最终(在一个扫描周期结束时)仍然需要标记对象，但我们会跟踪每个页面本地标记的对象，而不是跟踪整个堆中的标记对象。</li>
</ul>
<h3>绿茶示例</h3>
<p>让我们通过再次审视我们的示例堆，来看看这在实践中意味着什么，但这次运行的是“绿茶”而不是直接的图泛洪。</p>
<p>和之前一样，请跟随带注释的幻灯片进行浏览。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-060.png" alt="" /></p>
<p>这和之前的堆是一样的，但现在每个对象有两个比特的元数据而不是一个。同样，每个比特或框，对应于页中的一个对象槽位。总的来说，我们现在有 14 个比特对应于页 A 中的七个槽位。</p>
<p>顶部的比特代表和以前一样的东西：我们是否见过一个指向该对象的指针。我称之为“已见” (seen) 位。底部的比特集是新的。这些“已扫描” (scanned) 位跟踪我们是否已经扫描了该对象。</p>
<p>这块新的元数据是必需的，因为在“绿茶”中，<strong>工作列表跟踪的是页，而不是对象</strong>。我们仍然需要在某种程度上跟踪对象，这就是这些比特的目的。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-062.png" alt="" /></p>
<p>我们和以前一样开始，从根开始遍历对象。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-063.png" alt="" /></p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-064.png" alt="" /></p>
<p>但这一次，我们不是把一个对象放到工作列表上，而是把整个页——在这里是页 A——放到工作列表上，通过将整个页用蓝色阴影表示。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-066.png" alt="" /></p>
<p>我们找到的对象也是蓝色的，表示当我们从工作列表中取出这个页时，我们将需要查看那个对象。请注意，对象的蓝色调直接反映了页 A 中的元数据。其对应的“已见”位被设置，但其“已扫描”位没有。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-069.png" alt="" /></p>
<p>我们跟随下一个根，找到另一个对象，再次将整个页——页 C——放到工作列表上，并设置该对象的“已见”位。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-071.png" alt="" /></p>
<p>我们处理完根了，所以我们转向工作列表，并从工作列表中取出页 A。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-072.png" alt="" /></p>
<p>通过“已见”和“已扫描”位，我们可以知道页 A 上有一个对象需要扫描。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-074.png" alt="" /></p>
<p>我们扫描那个对象，跟随它的指针。结果，我们将页 B 添加到工作列表，因为页 A 中的第一个对象指向了页 B 中的一个对象。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-075.png" alt="" /></p>
<p>我们处理完页 A 了。接下来我们从工作列表中取出页 C。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-076.png" alt="" /></p>
<p>与页 A 类似，页 C 上有一个单独的对象需要扫描。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-078.png" alt="" /></p>
<p>我们在页 B 中找到了一个指向另一个对象的指针。页 B 已经在工作列表上了，所以我们不需要向工作列表添加任何东西。我们只需为目标对象设置“已见”位。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-079.png" alt="" /></p>
<p>现在轮到页 B 了。我们在页 B 上累积了两个待扫描的对象，我们可以按内存顺序，连续处理这两个对象！</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-081.png" alt="" /></p>
<p>我们遍历第一个对象的指针…</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-082.png" alt="" /></p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-083.png" alt="" /></p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-084.png" alt="" /></p>
<p>我们在页 A 中找到了一个指向一个对象的指针。页 A 之前在工作列表上，但此时不在了，所以我们把它放回工作列表。与原始的标记-清除算法不同，在原始算法中，任何给定的对象在整个标记阶段最多只会被添加到工作列表一次；而在“绿茶”中，一个给定的页在标记阶段可能会多次出现在工作列表上。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-085.png" alt="" /></p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-086.png" alt="" /></p>
<p>我们在扫描完第一个之后，立即扫描页中的第二个“已见”对象。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-087.png" alt="" /></p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-088.png" alt="" /></p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-089.png" alt="" /></p>
<p>我们在页 A 中又找到了几个对象…</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-090.png" alt="" /></p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-091.png" alt="" /></p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-092.png" alt="" /></p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-093.png" alt="" /></p>
<p>我们扫描完页 B 了，所以我们从工作列表中取出页 A。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-094.png" alt="" /></p>
<p>这次我们只需要扫描三个对象，而不是四个，因为我们已经扫描过第一个对象了。我们通过查看“已见”和“已扫描”位之间的差异，来知道要扫描哪些对象。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-095.png" alt="" /></p>
<p>我们将按顺序扫描这些对象。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-096.png" alt="" /></p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-097.png" alt="" /></p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-098.png" alt="" /></p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-099.png" alt="" /></p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-100.png" alt="" /></p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-101.png" alt="" /></p>
<p>我们完成了！工作列表上没有更多的页了，我们也没有正在处理的东西。请注意，现在元数据都很好地对齐了，因为所有可达的对象都既被“已见”又被“已扫描”。</p>
<p>你可能在我们的遍历过程中也注意到了，工作列表的顺序与图遍历有点不同。图遍历是“后进先出”或类似栈的顺序，而这里我们对工作列表上的页使用的是“先进先出”或类似队列的顺序。</p>
<p>这是有意为之的。当页在队列中等待时，我们让“已见”对象在每个页上累积，这样我们就可以一次性处理尽可能多的对象。这就是我们能一次性处理页 A 上那么多对象的原因。有时候，懒惰是一种美德。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-102.png" alt="" /></p>
<p>最后，我们可以像以前一样，清除掉未访问的对象。</p>
<h3>驶上高速公路</h3>
<p>让我们回到我们开车的比喻。我们终于要上高速公路了吗？</p>
<p>让我们回顾一下之前的图泛洪图片。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/graphflood-path2.png" alt="" /></p>
<p>原始图遍历在堆中穿行的路径需要 7 次独立的扫描。</p>
<p>我们到处跳跃，在不同的地方做着零碎的工作。“绿茶”所走的路径看起来非常不同。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/greentea-path.png" alt="" /></p>
<p>“绿茶”所走的路径仅需要 4 次扫描。</p>
<p>相比之下，绿茶在 A 和 B 页面上从左到右的移动次数较少，但每次移动时间更长。 这些箭头越长越好，箭头堆积越多，这种效果就越强。这就是绿茶的魅力所在。</p>
<p>这也是我们驰骋高速公路的机会。</p>
<p>这一切都使得它与微架构更加契合。现在，我们可以更精确地扫描彼此靠近的对象，从而更有可能利用缓存并避免使用主内存。同样，每页的元数据也更有可能被缓存。跟踪页面而非对象意味着工作列表更小，而工作列表压力的降低意味着争用更少，CPU 停顿也更少。</p>
<p>说到高速公路，我们可以把我们比喻意义上的引擎开到以前从未开过的档位，因为现在我们可以使用向量硬件了！</p>
<h3>向量加速</h3>
<p>如果你对向量硬件只有粗浅的了解，可能会不明白我们在这里如何使用它。但除了常见的算术和三角运算之外，最新的向量硬件还支持两项对绿茶算法非常有用的功能：超宽寄存器和复杂的位运算。</p>
<p>大多数现代 x86 CPU 都支持 AVX-512 指令集，它拥有 512 位宽的向量寄存器。如此宽的寄存器足以在 CPU 上仅使用两个寄存器来存储整个页面的所有元数据，从而使 Green Tea 能够仅用几条直线指令就完成整个页面的扫描。向量硬件长期以来一直支持对整个向量寄存器进行基本的位运算，但从 AMD Zen 4 和 Intel Ice Lake 开始，它还支持一种新的位向量“瑞士军刀”指令，使得 Green Tea 扫描过程中的关键步骤能够在几个 CPU 周期内完成。这些改进共同作用，使我们能够大幅提升 Green Tea 的扫描循环速度。</p>
<p>对于之前的图泛洪来说，这根本不可能，因为我们需要在各种大小的对象之间来回扫描。有时只需要两条元数据，有时却需要一万条。向量硬件根本无法满足这种可预测性和规律性要求。</p>
<p>如果你想深入了解一些细节，请继续阅读！否则，请随时跳到下面的【评估】小节。</p>
<h4>AVX-512 扫描内核</h4>
<p>要了解 AVX-512 GC 扫描是什么样子，请看下面的图。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/avx512.png" alt="" /><br />
<center>用于扫描的 AVX-512 矢量内核</center></p>
<p>这里面涉及的内容很多，我们可能光是解释它的运作原理就能写一整篇博客文章。现在，我们先从宏观层面来概括一下：</p>
<ol>
<li>首先，我们获取页面的“已查看”和“已扫描”位。请记住，页面中的每个对象对应一位，并且页面中的所有对象大小相同。 </li>
<li>接下来，我们比较这两个位集。它们的并集成为新的“扫描”位，而它们的差集则是“活动对象”位图，它告诉我们在本次页面扫描过程中（与之前的扫描相比）需要扫描哪些对象。</li>
<li>我们计算两个位图的差值并进行“扩展”，这样就不是每个对象占用一位，而是页面中的每个字（8 字节）占用一位。我们称之为“活动字”位图。例如，如果页面存储 6 个字（48 字节）的对象，则活动对象位图中的每位将被复制到活动字位图中的 6 位。如下所示：</li>
</ol>
<pre><code>0 0 1 1 ...  → 000000 000000 111111 111111 ...
</code></pre>
<ol>
<li>接下来，我们获取页面的指针/标量位图。同样，这里的每一位都对应页面的一个字（8 字节），并告诉我们该字是否存储指针。这些数据由内存分配器管理。</li>
<li>
<p>现在，我们取指针/标量位图和活动字位图的交集。结果就是“活动指针位图”：该位图告诉我们尚未扫描的任何活动对象中包含的整个页面中每个指针的位置。</p>
</li>
<li>
<p>最后，我们可以遍历页面内存并收集所有指针。逻辑上，我们遍历活动指针位图中的每个置位，加载该字处的指针值，并将其写回缓冲区。该缓冲区稍后将用于标记已访问的对象并将页面添加到工作列表中。利用向量指令，我们只需几条指令即可一次处理 64 字节。</p>
</li>
</ol>
<p>让这一切变快的部分原因是 VGF2P8AFFINEQB 指令，它是“Galios Field新指令” x86 扩展的一部分，也是我们上面提到的位操作“瑞士军刀”。它是真正的明星，因为它让我们能够非常高效地完成扫描内核中的第 (3) 步。它执行逐位的<a href="https://en.wikipedia.org/wiki/Affine_transformation">仿射变换</a>，将向量中的每个字节本身视为一个 8 位的数学向量，并将其与一个 8&#215;8 的比特矩阵相乘。这一切都是在<a href="https://en.wikipedia.org/wiki/Finite_field">Galios Field</a> GF(2) 上完成的，这意味着乘法是AND，加法是XOR。这样做的好处是，我们可以为每个对象大小定义几个 8&#215;8 的比特矩阵，来精确地执行我们需要的 1:n 比特扩展。</p>
<p>完整的汇编代码，请看<a href="https://cs.opensource.google/go/go/+/master:src/internal/runtime/gc/scan/scan_amd64.s;l=23;drc=041f564b3e6fa3f4af13a01b94db14c1ee8a42e0">这个文件</a>。“扩展器”为每个大小类别使用不同的矩阵和不同的排列，所以它们在一个由<a href="https://cs.opensource.google/go/go/+/master:src/internal/runtime/gc/scan/mkasm.go;drc=041f564b3e6fa3f4af13a01b94db14c1ee8a42e0">代码生成器</a>编写的<a href="https://cs.opensource.google/go/go/+/master:src/internal/runtime/gc/scan/expand_amd64.s;drc=041f564b3e6fa3f4af13a01b94db14c1ee8a42e0">单独文件</a>中。除了扩展函数，代码量其实不多。大部分代码都被极大地简化了，因为我们可以在纯粹位于寄存器中的数据上执行大部分上述操作。而且，希望很快这段汇编代码<a href="https://go.dev/issue/73787">将被 Go 代码所取代</a>！</p>
<p>感谢 Austin Clements 设计了这个过程。它非常酷，而且非常快！</p>
<h3>评估</h3>
<p>那么，这就是Green Tea的工作原理。它到底有多大帮助呢？</p>
<p>效果可能相当显著。即使不考虑向量增强，我们的基准测试套件也显示垃圾回收的 CPU 成本降低了 10% 到 40%。例如，如果应用程序 10% 的时间都花在了垃圾回收器上，那么根据工作负载的具体情况，整体 CPU 消耗将降低 1% 到 4%。垃圾回收 CPU 时间降低 10% 大致是典型的改进幅度。<br />
（有关这些细节，请参阅 <a href="https://go.dev/issue/73581">GitHub issue</a>。）</p>
<p>我们在谷歌内部推广了绿茶，并且大规模推广后也看到了类似的效果。</p>
<p>我们仍在推出向量增强功能，但基准测试和早期结果表明，这将额外带来 10%的 GC CPU 降低。</p>
<p>虽然大多数工作负载都能在一定程度上受益，但也有一些工作负载不会受益。</p>
<p>Green Tea 算法基于这样的假设：我们可以一次性在单页上累积足够多的对象进行扫描，从而抵消累积过程的成本。如果堆结构非常规则（对象大小相同，且在对象图中的深度也相近），那么这个假设显然成立。但是，有些工作负载通常要求我们每次只能扫描一个对象。这可能比图泛洪更糟糕，因为我们可能在尝试累积对象到页面上的过程中，反而做了更多工作，最终却失败了。</p>
<p>Green Tea 算法针对仅包含单个待扫描对象的页面进行了特殊处理。这有助于减少性能回退，但并不能完全消除它们。</p>
<p>然而，要超越图泛洪算法，所需的单页累积数据量远比你想象的要少。这项研究的一个意外发现是，每次仅扫描页面 2% 的数据就能取得比图泛洪算法更好的性能。</p>
<h3>可用性</h3>
<p>“绿茶”已经在最近的 Go 1.25 版本中作为实验性功能提供，并且可以通过在构建时将环境变量 GOEXPERIMENT 设置为 greenteagc 来启用。这不包括前述的向量加速。</p>
<p>我们预计在 Go 1.26 中将“绿茶”作为默认的垃圾回收器，但你仍然可以通过 GOEXPERIMENT=nogreenteagc 在构建时选择退出。Go 1.26 还将在较新的 x86 硬件上增加向量加速，并根据我们收集的反馈包含一系列的调整和改进。</p>
<p>如果可以，我们鼓励你尝试<a href="https://tonybai.com/2024/11/15/install-gotip-using-go-repo-mirror/">使用 Go 的最新tip版本</a>！如果你更喜欢使用 Go 1.25，我们也同样欢迎您的反馈。请参阅<a href="https://go.dev/issue/73581#issuecomment-2847696497">这个 GitHub 评论</a>，其中包含一些关于我们感兴趣的诊断信息、如果你可以分享的话，以及首选的反馈渠道的细节。</p>
<h2>旅程</h2>
<p>在结束这篇博文之前，让我们花点时间谈谈我们走到今天的历程，以及这项技术背后的人的因素。</p>
<p>绿茶的核心理念看似简单，就像某个人灵光一闪的灵感火花。</p>
<p>但事实并非如此。“绿茶”是许多人多年来共同努力和构思的成果。Go 团队的多位成员都参与了构思，包括 Michael Pratt、Cherry Mui、David Chase 和 Keith Randall。当时在英特尔工作的 Yves Vandriessche 的微架构见解也对设计探索起到了至关重要的作用。为了使这个看似简单的理念得以实现，我们尝试了许多方法，也处理了许多细节问题。</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/timeline.png" alt="" /><br />
<center>时间线描绘了我们在达到今天这种状态之前，尝试过的一些类似想法</center></p>
<p>这个想法的萌芽可以追溯到2018年。有趣的是，团队里的每个人都认为最初的想法是别人提出的。</p>
<p>绿茶这个名字是在2024年得来的。当时，奥斯汀在日本四处寻觅咖啡馆，喝了无数抹茶，并由此构思出了早期版本的原型！这个原型证明了绿茶的核心理念是可行的。从此，我们便开始了绿茶的研发之路。</p>
<p>在 2025 年，随着 Michael 将绿茶项目实施并投入生产，其理念进一步发展和变化。</p>
<p>这需要大量的协作探索，因为绿茶算法不仅仅是一个算法，而是一个完整的设计空间。我们认为，单凭我们中的任何一个人都无法独自驾驭它。仅仅有想法是不够的，你还需要弄清楚细节并加以验证。现在我们已经做到了，终于可以开始迭代了。</p>
<p>“绿茶”的未来是光明的。</p>
<p>再次，请通过设置 GOEXPERIMENT=greenteagc 来尝试它，并让我们知道它的效果如何！我们对这项工作感到非常兴奋，并希望听到你的声音！</p>
<hr />
<p>你的Go技能，是否也卡在了“熟练”到“精通”的瓶颈期？</p>
<ul>
<li>想写出更地道、更健壮的Go代码，却总在细节上踩坑？</li>
<li>渴望提升软件设计能力，驾驭复杂Go项目却缺乏章法？</li>
<li>想打造生产级的Go服务，却在工程化实践中屡屡受挫？</li>
</ul>
<p>继《<a href="http://gk.link/a/10AVZ">Go语言第一课</a>》后，我的《<a href="http://gk.link/a/12yGY">Go语言进阶课</a>》终于在极客时间与大家见面了！</p>
<p>我的全新极客时间专栏 《<a href="http://gk.link/a/12yGY">Tony Bai·Go语言进阶课</a>》就是为这样的你量身打造！30+讲硬核内容，带你夯实语法认知，提升设计思维，锻造工程实践能力，更有实战项目串讲。</p>
<p>目标只有一个：助你完成从“Go熟练工”到“Go专家”的蜕变！ 现在就加入，让你的Go技能再上一个新台阶！</p>
<p><img src="https://tonybai.com/wp-content/uploads/course-card/iamtonybai-banner-2.gif" alt="" /></p>
<hr />
<p><strong>想系统学习Go，构建扎实的知识体系？</strong></p>
<p>我的新书《<a href="https://book.douban.com/subject/37499496/">Go语言第一课</a>》是你的首选。源自2.4万人好评的极客时间专栏，内容全面升级，同步至Go 1.24。首发期有专属五折优惠，不到40元即可入手，扫码即可拥有这本300页的Go语言入门宝典，即刻开启你的Go语言高效学习之旅！</p>
<p><img src="https://tonybai.com/wp-content/uploads/2025/go-primer-published-4.png" alt="" /></p>
<hr />
<p>商务合作方式：撰稿、出书、培训、在线课程、合伙创业、咨询、广告合作。如有需求，请扫描下方公众号二维码，与我私信联系。</p>
<p><img src="http://image.tonybai.com/img/tonybai/iamtonybai-wechat-qr.png" alt="" /></p>
<p style='text-align:left'>&copy; 2025, <a href='https://tonybai.com'>bigwhite</a>. 版权所有. </p>
]]></content:encoded>
			<wfw:commentRss>https://tonybai.com/2025/10/31/deep-into-go-green-tea-gc/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
	</channel>
</rss>
