news 2026/4/16 14:03:54

Go1.26 新特性:10年,Goroutine 调度指标终于来了!

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Go1.26 新特性:10年,Goroutine 调度指标终于来了!

大家好,我是煎鱼。

今天给大家分享的是 Go1.26 即将引入的一个重磅特性—— Goroutine 调度指标。

这个特性看似不起眼,但对生产环境的可观测性来说,非常重要。值得升级。

背景

说起 Go 的 runtime/metrics 包,相信做过性能调优的同学都不陌生。

这个包从 Go1.16 就开始提供各种运行时的统计数据,像内存分配、GC 情况等。

很无奈的是,这么多年过去了,关于 Goroutine 调度器的 metrics 指标一直是空白的。

一些痛点

我们来看看实际开发中的痛点。

如果线上服务突然变慢了,打开 Grafana 监控一看:

  • CPU 使用率正常。

  • 内存也没问题。

  • GC 次数也在合理范围。

问题出在哪?

很可能是 Goroutine 调度出了问题,从而引发了一些异常点。 但你没法直接观测到:

  • 到底创建了多少个 Goroutine?

  • 有多少 Goroutine 在排队等待执行?

  • 有没有 Goroutine 卡在系统调用里出不来?

  • 线程数是不是爆了?

“新” 提案

这就很尴尬了。早在 2016 年,就有开发者在 issue #15490 中提出这个需求。在即将 2026 年倒是实现了。

描述的需求很简单:“MemStats 提供了一种监控内存分配和 GC 的方式,我们也需要一个类似的工具来监控 Goroutine 调度器。”

简单来讲,提案的诉求就是需要能看到:

  • 总共创建了多少 Goroutine。

  • 当前有多少 Goroutine。

  • 启动了多少线程。

  • Goroutine 从就绪到运行的延迟情况。

这个提案一提就是快 10 年,Go 核心团队终于在 Go1.26 要把它落地了。

新的 Goroutine 调度指标

接下来我们看下具体新增了哪些指标。

Go1.26 在 runtime/metrics 包中新增了 6 个关于 Goroutine 和线程的指标:

  1. 从程序启动到现在总共创建的 Goroutine 数量:/sched/goroutines-created:goroutines

  2. 处于系统调用或 cgo 调用中的 Goroutine 数量(近似值):/sched/goroutines/not-in-go:goroutines

  3. 已就绪但还没执行的 Goroutine 数量(近似值):/sched/goroutines/runnable:goroutines

  4. 正在执行的 Goroutine 数量(近似值):/sched/goroutines/running:goroutines

  5. 等待某个资源的 Goroutine 数量(近似值),例如等 I/O:/sched/goroutines/waiting:goroutines

  6. 当前 Go runtime 拥有的活跃线程数:/sched/threads/total:threads

这里有几个点需要注意:

1、按状态统计的 Goroutine 数量都标注了近似值。

为什么是近似的?因为 Goroutine 的状态切换非常频繁,Go runtime 不可能为了统计精确值而加锁,那样会严重影响性能。

所以这些数字可能有一点点偏差,但对监控来说已经足够了。

2、按状态分类的数量加起来,不一定等于当前存活的 Goroutine 总数(/sched/goroutines:goroutines)。这也是因为采样时机的问题。

3、所有指标都使用 uint64 类型的计数器。

使用例子

后面新版本发布后,我们来看下怎么用这些新指标。

假设我们写了一个简单的程序,启动了一些 Goroutine 做各种事情:

package main import ( "fmt" "runtime/metrics" "time" ) func main() { // 启动一些 Goroutine 模拟真实场景 go work() // 等待 100ms 让 Goroutine 跑一会儿 time.Sleep(100 * time.Millisecond) // 打印 Goroutine 相关指标 fmt.Println("Goroutine 调度指标:") printMetric("/sched/goroutines-created:goroutines", "累计创建") printMetric("/sched/goroutines:goroutines", "当前存活") printMetric("/sched/goroutines/not-in-go:goroutines", "系统调用/CGO") printMetric("/sched/goroutines/runnable:goroutines", "等待执行") printMetric("/sched/goroutines/running:goroutines", "正在执行") printMetric("/sched/goroutines/waiting:goroutines", "等待资源") // 打印线程相关指标 fmt.Println("\n线程指标:") printMetric("/sched/gomaxprocs:threads", "最大 P 数量") printMetric("/sched/threads/total:threads", "当前线程数") } func printMetric(name string, descr string) { sample := []metrics.Sample{{Name: name}} metrics.Read(sample) // 这里为了演示简化了错误处理 // 实际生产环境要检查 sample[0].Value.Kind fmt.Printf(" %s: %v\n", descr, sample[0].Value.Uint64()) } func work() { // 这里省略具体的工作逻辑 // 可能是网络请求、计算任务等 }

运行后输出大概是这样:

Goroutine 调度指标: 累计创建: 52 当前存活: 12 系统调用/CGO: 0 等待执行: 0 正在执行: 4 等待资源: 8 线程指标: 最大 P 数量: 8 当前线程数: 4

可以看到,程序累计创建了 52 个 Goroutine,当前还存活 12 个。

其中 4 个正在执行,8 个在等待某些资源(可能是 channel、锁、或者 I/O),没有 Goroutine 卡在系统调用里,也没有 Goroutine 在排队等待执行。

读取这些指标的方式和之前的runtime/metrics完全一样,都是通过metrics.Read来获取。使用上基本没有难度。

总结

这个提案从 2016 年提出到现在,社区一直呼声很高。

Go 核心团队成员 Michael Knyszek 最终在 2024 年底推动了这个特性的实现,相关的 CL(Change List)已经提交。

这个特性虽然 API 很简单,但对生产环境的可观测性意义重大。有了这些指标,我们终于能直接观测到 Goroutine 的调度状态,不用再猜来猜去了。

等 Go1.26 正式发布后,大家可以第一时间把这些指标用起来,相信会对大家定位生产问题有不小的帮助。

关注和加煎鱼微信,

一手消息和知识,拉你进技术交流群👇

你好,我是煎鱼,出版过 Go 畅销书《Go 语言编程之旅》,再到获得 GOP(Go 领域最有观点专家)荣誉,点击蓝字查看我的出书之路

日常分享高质量文章,输出 Go 面试、工作经验、架构设计,加微信拉读者交流群,和大家交流!

原创不易 点赞支持

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

如何通过两个小习惯提高我的数据科学家工作效率

原文:towardsdatascience.com/how-i-improved-my-productivity-as-a-data-scientist-with-two-small-habits-de09854d553c?sourcecollection_archive---------0-----------------------#2024-10-31 如何将这些习惯融入到日常生活中 https://ostiguyphilippe.mediu…

作者头像 李华
网站建设 2026/4/15 18:05:46

Update4j:构建Java应用自动更新的终极指南

Update4j:构建Java应用自动更新的终极指南 【免费下载链接】update4j Create your own auto-update framework 项目地址: https://gitcode.com/gh_mirrors/up/update4j 在当今快速迭代的软件开发环境中,保持应用程序的最新状态是提升用户体验的关…

作者头像 李华
网站建设 2026/4/11 8:33:35

Langchain-Chatchat能否支持多语言文档处理?

Langchain-Chatchat能否支持多语言文档处理? 在企业知识管理日益复杂的今天,一个常见的现实挑战浮出水面:如何让一份包含中、英、法、德等多种语言的技术文档集变得“可对话”?用户希望用中文提问,却能准确检索到英文报…

作者头像 李华
网站建设 2026/4/15 11:06:35

Realtek 8811CU驱动下载与安装指南:Windows 7无线网卡稳定配置方案

Realtek 8811CU驱动下载与安装指南:Windows 7无线网卡稳定配置方案 【免费下载链接】Realtek8811CU-21CU无线网卡驱动下载 Realtek 8811CU/8821CU无线网卡驱动专为Windows 7系统设计,支持802.11ac USB NIC,确保在AD-HOC模式下稳定运行。该驱动…

作者头像 李华