登录
首页 >  Golang >  Go教程

Golang执行Shell命令方法详解

时间:2026-02-04 20:08:39 416浏览 收藏

欢迎各位小伙伴来到golang学习网,相聚于此都是缘哈哈哈!今天我给大家带来《Golang执行Shell命令教程》,这篇文章主要讲到等等知识,如果你对Golang相关的知识非常感兴趣或者正在自学,都可以关注我,我会持续更新相关文章!当然,有什么建议也欢迎在评论留言提出!一起学习!

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

如何在Golang中使用os/exec包执行Shell命令_Golang Shell命令执行与输出获取

直接执行命令并获取标准输出

exec.Command 启动进程后,必须显式调用 Output()CombinedOutput() 才能拿到输出;只调用 Run() 不会返回任何内容,且 stdout/stderr 会被丢弃。

常见错误是写成:cmd.Run(); fmt.Println(cmd.Stdout)——Stdout 字段只是个 io.Writer 接口,没绑定实际缓冲区,打印出来是空的。

  • 想获取 stdout:用 cmd.Output(),它返回 []byteerror
  • 想同时捕获 stdout + stderr:用 cmd.CombinedOutput()
  • 需要实时处理流(比如日志滚动):手动设置 cmd.Stdout = &bufbytes.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 实例无效。

工作目录同理: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 -ttail -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/shbash,并把整条命令作为参数传入:

  • 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 展开。

到这里,我们也就讲完了《Golang执行Shell命令方法详解》的内容了。个人认为,基础知识的学习和巩固,是为了更好的将其运用到项目中,欢迎关注golang学习网公众号,带你了解更多关于的知识点!

前往漫画官网入口并下载 ➜
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>