确定Stdin是否有Go数据

确定Stdin是否有Go数据,go,pipe,stdin,Go,Pipe,Stdin,有没有办法检查输入流(os.Stdin)是否有数据 这篇文章展示了如何读取数据,但不幸的是,如果没有数据通过管道传输到stdin中,就会被阻塞。os.stdin与任何其他“文件”类似,因此您可以检查它的大小: package main import ( "fmt" "os" ) func main() { file := os.Stdin fi, err := file.Stat() if err != nil { fmt.Printl

有没有办法检查输入流(
os.Stdin
)是否有数据


这篇文章展示了如何读取数据,但不幸的是,如果没有数据通过管道传输到stdin中,就会被阻塞。

os.stdin与任何其他“文件”类似,因此您可以检查它的大小:

package main

import (
    "fmt"
    "os"
)

func main() {
    file := os.Stdin
    fi, err := file.Stat()
    if err != nil {
        fmt.Println("file.Stat()", err)
    }
    size := fi.Size()
    if size > 0 {
        fmt.Printf("%v bytes available in Stdin\n", size)
    } else {
        fmt.Println("Stdin is empty")
    }
}
我将其构建为“管道”可执行文件,其工作原理如下:

$ ./pipe
Stdin is empty
$ echo test | ./pipe
5 bytes available in Stdin

这似乎是一个可靠的解决方案,甚至可以通过管道处理睡眠/延迟数据。


一旦stdin中有可用数据,您将采取什么措施?我认为最好启动一个阻止输入的goroutine。我不相信这是正确的。在我看来,这取决于fstat在管道上的行为。我找不到任何参考资料,但根据经验,我会说posix没有很好地说明这一点。例如,如果stdin上的可用字节数超过管道缓冲大小(Linux上为4096),那么它会返回4096吗?思维实验:如果写入管道的程序每秒发送10个字节会怎么样?当缓冲区填满时,fstat会返回10、20等吗?但是,当缓冲区已满且写入程序处于睡眠状态时,会发生什么情况?如果写入程序不立即写入数据,则这是不可靠的。例如
(sleep 1;echo“hello”)|/pipe
打印的
Stdin是空的
显然不是空的。@NickCraig-Wood我同意,我对你的答案投了赞成票。将这两个问题合并在一起会很好。
package main

import (
  "os"
  "fmt"
)

func main() {
  fi, err := os.Stdin.Stat()
  if err != nil {
    panic(err)
  }
  if fi.Mode() & os.ModeNamedPipe == 0 {
    fmt.Println("no pipe :(")
  } else {
    fmt.Println("hi pipe!")
  }
}