
本文介绍如何利用 go 标准库结合操作系统底层接口(如 getrusage)准确获取子进程的最大驻留集大小(maxrss),强调必须依赖 os 原生支持,而非纯 go 实现。
在 Go 中,os/exec 包本身不提供内存监控能力——它仅负责进程的创建与生命周期管理。要获取子进程的内存使用量(尤其是峰值物理内存占用,即 Maximum Resident Set Size, MaxRSS),必须借助操作系统提供的资源使用统计机制。Go 通过 ProcessState.SysUsage() 方法将底层 OS 的 rusage 结构体(POSIX)或等效信息暴露给开发者,这是最可靠、开销最低的方案。
以下是一个典型示例,适用于 Linux、macOS 等 POSIX 兼容系统:
package main
import (
"fmt"
"log"
"os/exec"
"syscall"
)
func main() {
cmd := exec.Command("sleep", "1") // 替换为实际待测命令
err := cmd.Run()
if err != nil {
log.Fatal(err)
}
// 安全地提取 SysUsage 并断言为 *syscall.Rusage
if usage, ok := cmd.ProcessState.SysUsage().(*syscall.Rusage); ok {
// 注意:单位因平台而异!Linux 返回 KB,macOS 返回字节(需验证)
fmt.Printf("MaxRSS: %d\n", usage.Maxrss)
} else {
log.Fatal("failed to extract rusage")
}
}⚠️ 关键注意事项:
- Maxrss 的单位不是跨平台统一的:Linux(glibc)通常以 KB 为单位,而 macOS/BSD 默认返回 字节;具体请查阅对应系统的 man 2 getrusage 文档。生产环境建议根据 runtime.GOOS 做适配处理。
- SysUsage() 仅在进程已退出后才可用(即 cmd.Wait() 或 cmd.Run() 返回后),且要求进程正常终止或被信号终止(cmd.ProcessState.Exited() 应为 true)。
- *syscall.Rusage 是非导出类型,类型断言失败会 panic,务必配合 ok 判断使用。
- Windows 不支持 getrusage,SysUsage() 在该平台返回 nil;若需 Windows 支持,应改用 psutil 类库或调用 GetProcessMemoryInfo Win32 API(需 cgo 或外部工具)。
✅ 总结:测量子进程内存用量本质上是操作系统职责,Go 提供了安全、轻量的桥梁。优先使用 SysUsage() + *syscall.Rusage 是最佳实践,但务必校验平台行为、单位一致性与错误边界。避免轮询 /proc/










