doulianxi0587 2013-11-21 23:29
浏览 135
已采纳

为什么在我的Go程序中使用stderr和ioutil.ReadAll获得“坏文件描述符”

The command "psql" should throw an error, and I am trying to read stderr and print it in the Go program. I use ioutil.ReadAll to read the data from stderr, and stdout.

Unfortunately it is not reading from stderr at all. ioutil.ReadAll returns an error, which is not the error I am expecting.

The error I get is

read |0: bad file descriptor

Here is the code.

package main

import (
        "fmt"
        "os/exec"
        "io/ioutil"
)

func main() {
        cmd := exec.Command("psql")
        stdout, err := cmd.StdoutPipe()
        if err != nil {
                fmt.Printf("Error: %s", err)
        }
        stderr, err := cmd.StderrPipe()
        if err != nil {
                fmt.Printf("Error: %s", err)
        }
        err = cmd.Start()
        if err != nil {
                fmt.Printf("Start error %s",err)
        }

        d := cmd.Wait()
        if d != nil {
                fmt.Println(d)
        }

        stdo,g := ioutil.ReadAll(stdout)
        stde,f := ioutil.ReadAll(stderr)

        if g != nil {
                fmt.Println(g)
        }

        if f !=nil {
                fmt.Println(f)
        }

        fmt.Printf("Standard err is %s 
", stde)
        fmt.Printf("Standard out is %s 
",stdo)
}
  • 写回答

1条回答 默认 最新

  • dsovc00684 2013-11-22 08:45
    关注

    I found that through experimentation that I am getting the error, due to the fact that I am calling

       stdo,g := ioutil.ReadAll(stdout)
       stde,f := ioutil.ReadAll(stderr)
    

    after

     d := cmd.Wait()
    

    so what happens is the stdout, stderr pipe get closed after the cmd.Wait() returns.

    Here are the code comments for the cmd.StderrPipe()

    // StderrPipe returns a pipe that will be connected to the command's
    // standard error when the command starts.
    // The pipe will be closed automatically after Wait sees the command exit.
    

    So obviously we can't read stdout and stderr after they get closed.

    We cannot read them before the command starts either. So we have to put them in between start and wait.

    Here is code that fixes that.

    package main
    
    import (
            "fmt"
            "os/exec"
            "io/ioutil"
    )
    
    func main() {
            cmd := exec.Command("psql")
            stdout, err := cmd.StdoutPipe()
            if err != nil {
                    fmt.Printf("Error: %s", err)
            }
            stderr, err := cmd.StderrPipe()
            if err != nil {
                    fmt.Printf("Error: %s", err)
            }
            err = cmd.Start()
            if err != nil {
                    fmt.Printf("Start error %s",err)
            }
    
            stdo,g := ioutil.ReadAll(stdout)
            stde,f := ioutil.ReadAll(stderr)
    
            d := cmd.Wait()
    
            if d != nil {
                    fmt.Println(d)
            }
    
            if g != nil {
                    fmt.Println(g)
            }
    
            if f !=nil {
                    fmt.Println(f)
            }
    
            fmt.Printf("Standard err is %s 
    ", stde)
            fmt.Printf("Standard out is %s 
    ",stdo)
    }
    
    本回答被题主选为最佳回答 , 对您是否有帮助呢?
    评论

报告相同问题?

悬赏问题

  • ¥15 file converter 转换格式失败 报错 Error marking filters as finished,如何解决?
  • ¥15 ubuntu系统下挂载磁盘上执行./提示权限不够
  • ¥15 Arcgis相交分析无法绘制一个或多个图形
  • ¥15 关于#r语言#的问题:差异分析前数据准备,报错Error in data[, sampleName1] : subscript out of bounds请问怎么解决呀以下是全部代码:
  • ¥15 seatunnel-web使用SQL组件时候后台报错,无法找到表格
  • ¥15 fpga自动售货机数码管(相关搜索:数字时钟)
  • ¥15 用前端向数据库插入数据,通过debug发现数据能走到后端,但是放行之后就会提示错误
  • ¥30 3天&7天&&15天&销量如何统计同一行
  • ¥30 帮我写一段可以读取LD2450数据并计算距离的Arduino代码
  • ¥15 飞机曲面部件如机翼,壁板等具体的孔位模型