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

问题描述 投票:0回答:3

当我运行下面的代码时:

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

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

如何获取更详细的信息?

error-handling command go exec
3个回答
155
投票

解决方案是使用 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:find:-exec:没有终止“;”或“+”

编辑:

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

但是,正如 @snorberhuis 下面提到的,某些命令会将错误打印到标准输出。其他命令可能会打印到 stderr 但返回错误代码 0(在这种情况下

err
将是
nil
)。 stderr 中包含消息并不一定意味着存在错误(ffmpeg 工具经常这样做)。

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


60
投票

正如 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


0
投票

可以在命令行后添加

;exit 0
强制返回0

cmd := exec.Command("find", "/", "-maxdepth", "1", "-exec", "wc", "-c", "{}", "\\", ";exit 0")
© www.soinside.com 2019 - 2024. All rights reserved.