用exec.Command执行命令需显式调用Output()或CombinedOutput()获取输出;Run()不返回内容且丢弃stdout/stderr;环境变量和工作目录须通过cmd.Env、cmd.Dir设置;超时需用CommandContext配合context;shell特性如管道、通配符需显式调用sh或cmd。

直接执行命令并获取标准输出
用 exec.Command 启动进程后,必须显式调用 Output() 或 CombinedOutput() 才能拿到输出;只调用 Run() 不会返回任何内容,且 stdout/stderr 会被丢弃。
常见错误是写成:cmd.Run(); fmt.Println(cmd.Stdout)——Stdout 字段只是个 io.Writer 接口,没绑定实际缓冲区,打印出来是空的。
- 想获取 stdout:用
cmd.Output(),它返回[]byte和error - 想同时捕获 stdout + stderr:用
cmd.CombinedOutput() - 需要实时处理流(比如日志滚动):手动设置
cmd.Stdout = &buf(bytes.Buffer)或os.Pipe()
示例:
cmd := exec.Command("ls", "-l")
out, err := cmd.Output()
if err != nil {
log.Fatal(err)
}
fmt.Println(string(out))
带环境变量和工作目录的命令执行
exec.Command 默认继承父进程环境,但若需覆盖或新增变量,得通过 cmd.Env 显式设置;直接改 os.Setenv 对已创建的 cmd 实例无效。
立即学习“go语言免费学习笔记(深入)”;
工作目录同理:cmd.Dir 必须在 Run() 前赋值,否则按当前进程目录执行。
- 继承当前环境并追加变量:
cmd.Env = append(os.Environ(), "PATH=/usr/local/bin:/usr/bin") - 完全自定义环境(不继承):
cmd.Env = []string{"HOME=/tmp", "LANG=en_US.UTF-8"} - 设置工作目录:
cmd.Dir = "/tmp/myproject" - 注意 Windows 下路径分隔符不影响
exec.Command,它内部会自动转换
处理命令超时与信号中断
exec.Command 本身不支持超时,必须配合 context.WithTimeout 使用;否则一个卡死的 ping -t 或 tail -f 会让整个 goroutine 挂住。
用 exec.CommandContext 替代 exec.Command,并在 context 被取消时,子进程会被 SIGKILL 终止(Unix)或 TerminateProcess(Windows)。
- 不要用
time.AfterFunc+cmd.Process.Kill(),竞态风险高 - 超时后检查
err类型:errors.Is(err, context.DeadlineExceeded)可区分超时和其他错误 - 若需发送 SIGINT(如模拟 Ctrl+C),用
cmd.Process.Signal(os.Interrupt),但要确保进程未结束
示例:
ctx, cancel := context.WithTimeout(context.Background(), 3*time.Second)
defer cancel()
cmd := exec.CommandContext(ctx, "sleep", "5")
err := cmd.Run()
if errors.Is(err, context.DeadlineExceeded) {
fmt.Println("command timed out")
}
Shell 特性(管道、重定向、通配符)不能直接用
exec.Command("ls | grep main") 会报错 “executable file not found”,因为 Go 不调用 shell 解析器,默认执行的是字面量程序名。管道 |、重定向 >、通配符 * 都属于 shell 功能。
真要依赖 shell 行为,得显式调用 /bin/sh 或 bash,并把整条命令作为参数传入:
- Unix:
exec.Command("/bin/sh", "-c", "ls *.go | grep main") - Windows:
exec.Command("cmd", "/C", "dir *.go ^| findstr main")(注意^转义管道) - 但这样做牺牲了可移植性、安全性(注意命令注入)和性能,应优先拆解为多个
exec.Command并用管道连接
最易被忽略的一点:哪怕只是想展开 *,也得走 shell,exec.Command("ls", "*.go") 会原样传给 ls,由它自己处理——而很多工具并不做 glob 展开。










