Go语言监控系统采集教程详解
时间:2026-01-16 10:18:47 247浏览 收藏
来到golang学习网的大家,相信都是编程学习爱好者,希望在这里学习Golang相关编程知识。下面本篇文章就来带大家聊聊《Go语言监控程序实现与系统采集教程》,介绍一下,希望对大家的知识积累有所帮助,助力实战开发!
优先选 gopsutil 而非原生 syscall 或 /proc:它跨平台抽象了 Linux/cgroup v2、Windows/macOS 差异,但需注意 cpu.Percent() 必须传 time.Duration 才采样、disk.Usage() 在容器中需先用 disk.Partitions(true) 查真实挂载点、host.Info() 在 Alpine 中 fallback 到空字符串。

Go 语言写系统监控程序,核心不是“能不能”,而是“要不要自己造轮子”——gopsutil 已覆盖绝大多数场景,直接用它比手撸 /proc 解析或调用 sysctl 稳定得多。
为什么优先选 gopsutil 而不是原生 syscall 或 /proc
自己读 /proc/meminfo 或调用 runtime.ReadMemStats() 看内存,看似轻量,但很快会遇到:不同 Linux 发行版 /proc 字段顺序不一致、cgroup v2 下 /proc/cgroups 不再可用、Windows/macOS 完全不可用。而 gopsutil 在这些平台做了抽象和 fallback。
gopsutil的cpu.Percent()默认返回每核使用率,加time.Second参数才触发采样;漏传会导致返回全 0disk.Usage("/path")在容器中可能因挂载命名空间隔离失败,需配合disk.Partitions(true)先查真实挂载点- 它的
host.Info()依赖/etc/os-release,Alpine 镜像里默认没这个文件,会 fallback 到空字符串
采集 CPU/内存/磁盘时必须设对的参数
默认行为往往不是你想要的。比如 cpu.Percent() 不传 time.Duration 就不会真正采样,mem.VirtualMemory() 返回的是瞬时快照,不带历史趋势。
package main
import (
"fmt"
"time"
"github.com/shirou/gopsutil/cpu"
"github.com/shirou/gopsutil/mem"
"github.com/shirou/gopsutil/disk"
)
func main() {
// ✅ 正确:指定采样间隔,否则返回 []float64{}
percents, _ := cpu.Percent(time.Second, false)
fmt.Printf("CPU: %.1f%%\n", percents[0])
// ✅ 正确:获取当前内存使用
v, _ := mem.VirtualMemory()
fmt.Printf("Mem: %.1f%% used (%d/%d MB)\n", v.UsedPercent, v.Used/1024/1024, v.Total/1024/1024)
// ✅ 正确:先找根分区,避免硬编码 "/"
parts, _ := disk.Partitions(true)
for _, p := range parts {
if p.Mountpoint == "/" {
du, _ := disk.Usage(p.Mountpoint)
fmt.Printf("Disk /: %.1f%% used\n", du.UsedPercent)
break
}
}
}
HTTP 暴露指标时别直接用 net/http 返回 JSON
裸写 http.HandleFunc + json.Marshal 看似简单,但会快速撞上三个问题:并发采集冲突(多个请求同时调 cpu.Percent())、无超时控制(disk.Usage() 在 NFS 挂死时卡住整个 handler)、指标格式不兼容 Prometheus。
- 用
sync.Once或sync.RWMutex保护采集逻辑,避免重复初始化或并发读写结构体 - 所有
gopsutil调用必须包在context.WithTimeout里,例如ctx, cancel := context.WithTimeout(context.Background(), 3*time.Second) - 如果后续要对接 Prometheus,直接用
prometheus.NewGaugeVec+promhttp.Handler(),别自己拼/metrics文本
容器环境下的常见失效点
在 Docker/K8s 里跑 Go 监控程序,gopsutil 默认行为会误判宿主机资源。关键不是“能不能运行”,而是“它读的是哪一层的视图”。
- 容器内执行
host.Info()返回的是宿主机的hostname和uptime,但os.Getpid()是容器 PID,混用会导致日志混乱 process.Pids()默认列出所有进程,包括宿主机的,需过滤p.Pid > 0 && p.Pid < 1000000(容器 PID 通常较小)或读/proc/1/cgroup判断是否在容器中- 想查容器自身 CPU limit,得解析
/sys/fs/cgroup/cpu/cpu.cfs_quota_us,gopsutil不提供该接口,必须手动读
真正难的不是采集数据,而是明确“这个值在当前运行环境中代表什么”。比如 cpu.Times(false) 返回的 user 时间,在容器里是受限于 cgroup 的,还是宿主机全局的——得看你是从 /proc/stat 还是 /sys/fs/cgroup/cpu/xxx/cpu.stat 读的。
文中关于的知识介绍,希望对你的学习有所帮助!若是受益匪浅,那就动动鼠标收藏这篇《Go语言监控系统采集教程详解》文章吧,也可关注golang学习网公众号了解相关技术文章。
-
505 收藏
-
503 收藏
-
502 收藏
-
502 收藏
-
502 收藏
-
186 收藏
-
395 收藏
-
196 收藏
-
355 收藏
-
242 收藏
-
151 收藏
-
292 收藏
-
106 收藏
-
256 收藏
-
165 收藏
-
172 收藏
-
463 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习