在 Golang 中运行 exec.Command 时如何调试“退出状态 1”错误

Posted

技术标签:

【中文标题】在 Golang 中运行 exec.Command 时如何调试“退出状态 1”错误【英文标题】:How to debug "exit status 1" error when running exec.Command in Golang 【发布时间】:2013-08-12 03:41:38 【问题描述】:

当我运行下面的代码时:

cmd := exec.Command("find", "/", "-maxdepth", "1", "-exec", "wc", "-c", "", "\\")
var out bytes.Buffer
cmd.Stdout = &out
err := cmd.Run()
if err != nil 
    fmt.Println(err)
    return

fmt.Println("Result: " + out.String())

我收到此错误:

退出状态 1

但是这对于调试错误的确切原因没有帮助。

如何获取更详细的信息?

【问题讨论】:

【参考方案1】:

解决方案是使用 Command 对象的Stderr 属性。可以这样做:

cmd := exec.Command("find", "/", "-maxdepth", "1", "-exec", "wc", "-c", "", "\\")
var out bytes.Buffer
var stderr bytes.Buffer
cmd.Stdout = &out
cmd.Stderr = &stderr
err := cmd.Run()
if err != nil 
    fmt.Println(fmt.Sprint(err) + ": " + stderr.String())
    return

fmt.Println("Result: " + out.String())

运行上面的代码,可以清楚问题是什么:

退出状态1:查找:-exec:没有终止“;”或“+”

编辑:

在上面的代码中,我们期望在出错的情况下,消息将被打印到 stderr 并且命令将返回一个非零错误代码。这或多或少是标准的。

但是,正如@snorberhuis 下面提到的,一些命令会将错误打印到标准输出。其他命令可能会打印到 stderr 但返回错误代码 0(在这种情况下,err 将是 nil)。在 stderr 中有消息并不一定意味着有错误(ffmpeg 工具经常这样做)。

所以基本上你可能需要调整上面的代码以适应你期望的命令。

【讨论】:

您的解决方案对调试非常有帮助,但我必须做一些小改动才能真正看到输出。我也在使用退出状态 1 调试 Windows 关机命令。这将打印到 Stdout。所以我把最后一行打印移到 if 语句上面。 这个解决方案比CombinedOutput() 更好,因为在这里我们得到了详细的错误文本(cmd.Stderr)。在 output, err := cmd.CombinedOutput() 中,output 确实包含错误文本,但与 err 相同。【参考方案2】:

正如 Laurent 所提到的,您可以覆盖 Stderr 文件描述符以捕获 stderr 输出以获得更好的错误消息。如果做一些相对简单的事情,我个人更喜欢使用CombinedOutput 方法来执行命令:

cmd := exec.Command("find", "/", "-maxdepth", "1", "-exec", "wc", "-c", "", "\\")
output, err := cmd.CombinedOutput()
if err != nil 
    fmt.Println(fmt.Sprint(err) + ": " + string(output))
    return

fmt.Println(string(output))

这是上面示例的 play.golang.org 链接:http://play.golang.org/p/z8k9zO755P

【讨论】:

这个答案更优雅;D 完美答案:)

以上是关于在 Golang 中运行 exec.Command 时如何调试“退出状态 1”错误的主要内容,如果未能解决你的问题,请参考以下文章

防止 Ctrl+C 中断 Golang 中的 exec.Command

Golang os/exec 实现

在golang中处理逻辑错误与编程错误的惯用方法

Golang Linux Shell编程

Golang exec:stat:文件被移动到那里后没有这样的文件或目录

golang调用EXE