news 2026/6/23 19:08:22

大项目构建太慢?Brad Fitzpatrick 提议引入 -cachelink 降低测试等待时间

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
大项目构建太慢?Brad Fitzpatrick 提议引入 -cachelink 降低测试等待时间

大家好,我是Tony Bai。

在维护大型 Go 单体仓库(Monorepo)时,你是否遇到过这样的场景:明明只是修改了测试的运行参数(比如-run的正则),或者在不同的 CI 节点上运行同一个包的测试,却发现go test依然在缓慢地执行“链接(Linking)”步骤?

对于代码量巨大的项目,链接过程往往是构建链条中最耗时的一环。为了解决这一痛点,Go 社区领袖、Tailscale 核心开发者 Brad Fitzpatrick 近日提交了 #77349 提案,建议引入-cachelink标志。这一看似微小的改动,有望在分布式测试和重复执行场景下,显著“挤出”原本被浪费的等待时间。

被忽视的瓶颈:重复链接的代价

Go 的构建缓存(GOCACHE)机制已经非常高效,它能很好地缓存编译阶段的中间产物(.a文件)。但是,当你运行go test时,工具链的最后一步——将所有依赖链接成一个可执行的测试二进制文件——通常是“一次性”的。

这意味着,即使你的代码没有任何变动,只要测试指令稍有变化(例如多次运行go test但指定不同的测试用例),Go 工具链往往会重新触发链接器。

# 第一次运行:链接 + 执行 $ go test -run=^TestFoo$ ./pkg/ # 第二次运行(代码未变):依然触发重新链接 + 执行 $ go test -run=^TestBar$ ./pkg/

对于依赖项数以千计的大型项目,链接过程可能长达数秒甚至更久。在本地频繁调试或 CI 流水线中,这些重复的秒数累积起来就是巨大的时间浪费。

Brad 的解法:-cachelink

Brad Fitzpatrick 的提案非常直接:允许将链接器输出的最终测试二进制文件,也写入 GOCACHE。

通过显式开启-cachelinkgo test的行为将发生变化:

  1. 它会基于构建输入(代码、依赖、环境变量等)计算哈希。

  2. 如果发现 GOCACHE 中已经存在已链接好的测试二进制文件。

  3. 直接跳过链接步骤,复用该文件进行测试。

这样,上述例子中的第二次调用将瞬间启动,因为最耗时的构建步骤被完全省去了。

为什么不做成默认行为?

既然能提速,为什么不默认开启?Brad 在提案讨论中给出了专业的权衡分析:

空间 vs. 时间

测试二进制文件通常包含完整的符号表和调试信息,体积比普通的中间对象文件大得多。如果默认缓存所有测试二进制文件,开发者的磁盘空间(GOCACHE)会迅速膨胀。因此,这是一个以空间换时间的策略,更适合由开发者根据项目规模手动开启,或者在 CI 环境中配置。

分布式 CI 的“加速器”

该提案真正的杀手级应用场景是 分布式 CI 系统。

许多大厂使用GOCACHEPROG来在构建集群间共享缓存。在典型的 CI 流程中,测试任务往往会被分片(Sharding)到数十台机器上并发执行。

  • 现状:每一台机器拉取源码后,都需要各自进行一次链接操作,浪费计算资源。

  • 引入-cachelink后:第一台完成构建的机器会将二进制文件上传到共享缓存。后续几十台机器直接下载该文件并运行,全集群的链接成本降为“1”。

不仅是go test -c

有经验的开发者可能会问:“我为什么不直接用go test -c手动编译成二进制文件,然后分发运行呢?”

Brad 指出,手动管理二进制文件会绕过 Go 原生的测试结果缓存。而-cachelink的精妙之处在于,它既复用了二进制文件,又保留了go test完整的缓存与输出管理体验。你不需要编写复杂的脚本来管理这些文件,一切依然由go命令自动处理。

小结

目前,该提案已进入活跃评审阶段,并有了初步的代码实现。对于深受“构建慢”和“测试慢”困扰的大型项目维护者来说,这无疑是一个值得期待的性能优化利器。我们有望在 Go 1.27 或后续版本中见证它的落地。

资料链接:https://github.com/golang/go/issues/77349


聊聊你的构建之苦

链接时间正在成为你的“带薪摸鱼”理由吗?在你的项目中,go test运行一次通常需要多久?你为了缩短测试反馈周期,还尝试过哪些黑科技(比如GOCACHEPROG)?

欢迎在评论区分享你的实战经验或吐槽!让我们一起期待-cachelink的落地。👇


点击下面标题,干货!

- 深入GOCACHEPROG:Go构建缓存的自定义扩展

- Go 标准库竟然也用 vendor?std 和 cmd 模块是如何管理外部依赖的

- Go 模块构建与依赖管理:我们到底在“折腾”什么?

- 【Go 测试之道】01 开篇:测试的“道”与“术”——从“演员对台词”到我们的“短链接”蓝图

- Brad Fitzpatrick 也等不及了!sync.Map 的泛型进化与 sync/v2 的诞生之路

- 再见,丑陋的 container/heap!Go 泛型堆 heap/v2 提案解析

- Go 泛型落地 4 年后,终于要支持泛型方法了!


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

  • 想写出更地道、更健壮的Go代码,却总在细节上踩坑?

  • 渴望提升软件设计能力,驾驭复杂Go项目却缺乏章法?

  • 想打造生产级的Go服务,却在工程化实践中屡屡受挫?

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

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

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

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/23 15:54:37

RAG检索新利器:Qwen2.5-VL多模态语义评估引擎实战解析

RAG检索新利器:Qwen2.5-VL多模态语义评估引擎实战解析 在RAG系统落地过程中,你是否遇到过这些真实困境? 检索阶段召回了20个文档,但其中真正匹配用户意图的可能只有3个; 图文混合查询(比如“对比这张电路图…

作者头像 李华
网站建设 2026/6/23 15:50:21

Ollama部署GLM-4.7-Flash:30B最强模型5分钟快速上手教程

Ollama部署GLM-4.7-Flash:30B最强模型5分钟快速上手教程 你是不是也遇到过这样的情况:听说有个新模型性能超强,赶紧去查文档——结果第一步就卡在“环境配置”上?装Ollama、拉模型、配CUDA、调端口……折腾一小时,连“…

作者头像 李华
网站建设 2026/6/19 9:48:15

RMBG-2.0在艺术创作中的应用:数字绘画辅助工具开发

RMBG-2.0在艺术创作中的应用:数字绘画辅助工具开发 1. 当艺术家遇到抠图难题:为什么传统方法不再够用 数字绘画创作中,一个看似简单却反复消耗精力的环节常常让人头疼——把人物或物体从原始图片中干净利落地分离出来。很多插画师朋友跟我聊…

作者头像 李华
网站建设 2026/6/13 12:09:10

零基础入门:用LoRA训练助手轻松搞定Stable Diffusion标签

零基础入门:用LoRA训练助手轻松搞定Stable Diffusion标签 你是不是也遇到过这样的情况:辛辛苦苦画了一张角色设定图,准备做LoRA训练,结果卡在第一步——写英文训练标签上?翻词典、查社区、拼凑语法,折腾半…

作者头像 李华
网站建设 2026/6/20 9:31:56

零基础玩转GLM-4-9B-Chat-1M:200万字文档一键分析实战

零基础玩转GLM-4-9B-Chat-1M:200万字文档一键分析实战 你有没有试过把一份300页的PDF财报拖进对话框,却只得到“内容过长,请分段输入”的提示?有没有为了一份50页的技术合同反复粘贴、反复提问,最后还漏看了关键条款&…

作者头像 李华