确定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!")
}
}