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

lau*_*ent 37 error-handling command exec go

当我运行以下代码时:

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())
Run Code Online (Sandbox Code Playgroud)

我收到此错误:

退出状态1

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

如何获得更详细的信息?

lau*_*ent 84

解决方案是使用StderrCommand对象的属性.这可以这样做:

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())
Run Code Online (Sandbox Code Playgroud)

运行上面的代码,可以清楚地说明问题所在:

退出状态1:查找:-exec:无终止";" 或"+"

编辑:

在上面的代码中,我们希望在出现错误的情况下,消息将打印到stderr,命令将返回非零错误代码.这或多或少是标准的.

但是,如下面@snorberhuis所述,某些命令会将错误打印到stdout.其他命令可能会打印到stderr但返回错误代码0(在这种情况下err将是nil).在stderr中发送消息并不一定意味着存在错误(ffmpeg工具会做很多事情).

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

  • 这个解决方案比“CombinedOutput()”更好,因为在这里我们得到了详细的错误文本(cmd.Stderr)。在 `output, err := cmd.CombinedOutput()` 中, `output` 确实包含错误文本,但与 `err` 相同。 (2认同)

小智 28

正如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))
Run Code Online (Sandbox Code Playgroud)

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