Reading input from stdin in golang

huangapple go评论101阅读模式
英文:

Reading input from stdin in golang

问题

我有这段 Go 代码:

  1. func readTwoLines() {
  2. reader := bufio.NewReader(os.Stdin)
  3. line, _ := reader.ReadString('\n')
  4. fmt.Println(line)
  5. line, _ = reader.ReadString('\n')
  6. fmt.Println(line)
  7. }

对于输入:

  1. hello
  2. bye

输出是:

  1. hello
  2. bye

一切正常。但是,如果我为每一行创建一个读取器:

  1. func readTwoLines() {
  2. line, _ := bufio.NewReader(os.Stdin).ReadString('\n')
  3. fmt.Println(line)
  4. line, err := bufio.NewReader(os.Stdin).ReadString('\n')
  5. if err != nil {
  6. fmt.Println(err)
  7. }
  8. fmt.Println(line)
  9. }

在第二行读取时会出现 EOF 错误。

为什么会发生这种情况?

英文:

I have this go code:

  1. func readTwoLines() {
  2. reader := bufio.NewReader(os.Stdin)
  3. line, _ := reader.ReadString('\n')
  4. fmt.Println(line)
  5. line, _ = reader.ReadString('\n')
  6. fmt.Println(line)
  7. }

For the input:

  1. hello
  2. bye

the output is:

  1. hello
  2. bye

Everything ok. But now, if I create one reader per line:

  1. func readTwoLines() {
  2. line, _ := bufio.NewReader(os.Stdin).ReadString('\n')
  3. fmt.Println(line)
  4. line, err := bufio.NewReader(os.Stdin).ReadString('\n')
  5. if err != nil {
  6. fmt.Println(err)
  7. }
  8. fmt.Println(line)
  9. }

there is an EOF error, in the second line reading.

Why is it happening?

答案1

得分: 21

对于简单的用途,使用Scanner可能更方便。你不应该使用两个读取器,首先读取,缓冲区大小为4096字节:

  1. // NewReader返回一个新的Reader,其缓冲区具有默认大小。
  2. func NewReader(rd io.Reader) *Reader {
  3. return NewReaderSize(rd, defaultBufSize)
  4. }

其中defaultBufSize = 4096

即使你的输入包含4000字节,第二次读取仍然没有内容可读。但是,如果你输入的内容超过4096字节,它将起作用。

如果在找到分隔符之前,ReadString遇到错误,它将返回错误之前读取的数据和错误本身(通常是io.EOF)。

这是设计如此,参见文档:

  1. // ReadString读取输入中第一个分隔符之前的内容,返回包含数据和分隔符的字符串。
  2. // 如果在找到分隔符之前,ReadString遇到错误,它将返回错误之前读取的数据和错误本身(通常是io.EOF)。
  3. // 当且仅当返回的数据不以分隔符结尾时,ReadString返回err != nil。
  4. // 对于简单的用途,使用Scanner可能更方便。
  5. func (b *Reader) ReadString(delim byte) (string, error) {
  6. bytes, err := b.ReadBytes(delim)
  7. return string(bytes), err
  8. }

尝试这个:

  1. package main
  2. import (
  3. "bufio"
  4. "fmt"
  5. "os"
  6. )
  7. func main() {
  8. scanner := bufio.NewScanner(os.Stdin)
  9. for scanner.Scan() {
  10. fmt.Println(scanner.Text()) // Println会添加最后的'\n'
  11. }
  12. if err := scanner.Err(); err != nil {
  13. fmt.Fprintln(os.Stderr, "reading standard input:", err)
  14. }
  15. }

运行:

  1. go run m.go < in.txt

输出:

  1. hello
  2. bye

in.txt文件:

  1. hello
  2. bye

希望这可以帮到你。

英文:

For simple uses, a Scanner may be more convenient.
You should not use two readers, first read, buffers 4096 bytes of input:

> // NewReader returns a new Reader whose buffer has the default size.
> func NewReader(rd io.Reader) *Reader {
> return NewReaderSize(rd, defaultBufSize)
> }

and defaultBufSize = 4096

and even your input contains 4000 bytes, still second read got nothing to read.
but if you enter input more than 4096 bytes it will work.


> If ReadString encounters an error before finding a delimiter, it
> returns the data read before the error and the error itself (often
> io.EOF).

it is by design, see doc:

  1. // ReadString reads until the first occurrence of delim in the input,
  2. // returning a string containing the data up to and including the delimiter.
  3. // If ReadString encounters an error before finding a delimiter,
  4. // it returns the data read before the error and the error itself (often io.EOF).
  5. // ReadString returns err != nil if and only if the returned data does not end in
  6. // delim.
  7. // For simple uses, a Scanner may be more convenient.
  8. func (b *Reader) ReadString(delim byte) (string, error) {
  9. bytes, err := b.ReadBytes(delim)
  10. return string(bytes), err
  11. }

try this:

  1. package main
  2. import (
  3. &quot;bufio&quot;
  4. &quot;fmt&quot;
  5. &quot;os&quot;
  6. )
  7. func main() {
  8. scanner := bufio.NewScanner(os.Stdin)
  9. for scanner.Scan() {
  10. fmt.Println(scanner.Text()) // Println will add back the final &#39;\n&#39;
  11. }
  12. if err := scanner.Err(); err != nil {
  13. fmt.Fprintln(os.Stderr, &quot;reading standard input:&quot;, err)
  14. }
  15. }

run:

  1. go run m.go &lt; in.txt

output:

  1. hello
  2. bye

in.txt file:

  1. hello
  2. bye

I hope this helps.

huangapple
  • 本文由 发表于 2017年8月28日 15:39:56
  • 转载请务必保留本文链接:https://go.coder-hub.com/45914003.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定