EOF on Windows

最近在学习go时,在运行一个简单的读取标准输入的例子时发现,在控制台上(cmd)输入ctrl-z竟然不能结束程序。以前在学习c/c++时,也遇到过有关的问题,也就是当我想要结束输入时,如果在输入ctrl-z之前又输入了某些其他字符,那么此时标准输入不会被关闭,而当一行中仅有一个ctrl-z时,才能结束输入。下面一个简单的c++程序可以验证这个问题:

#include <iostream>
#include <string>
using namespace std;

int main()
{
    string line;
    while (getline(cin, line)) {
        cout << line << endl;
    }
}

如果我们运行时输入

1234^Z5678

此时如果只按一次回车,是看不到输出的,当我们再按一次回车时,才能看到如下输出

1234->

这里->是输出的一个箭头符号代表^Z,但是当我们输入一行仅包含^Z的字符时,程序会正常结束。这里可以看到

  1. 不在行首的^Z无法关闭标准输入,否则我们应该看到1234而不是上面的输出
  2. ^Z也会被当做一个普通字符
  3. ^Z所在行之后的字符(包括换行)都不会显示,也就是库函数会将^Z之后的该行字符全部丢掉

具体造成这种问题的原因,可以参考All About EOF,文中提到了使用ctrl-z作为伪EOF的历史原因,还提到了使用eof检测函数的一些常见错误。

*nix系统的终端默认使用ctrl-d来结束标准输入,但是ctrl-d其实也不是EOF,EOF只是在当标准输入关闭时,读取后返回的一个特殊值。ctrl-d会被终端解释为关闭标准输入的指令。

在windows的console上,我们可以如下检测^Z

package main

import (
    "bufio"
    "golang.org/x/crypto/ssh/terminal"
    "os"
    "runtime"
)

func main() {
    in := bufio.NewReader(os.Stdin)
    for {
        r, n, err := in.ReadRune()
        if runtime.GOOS == "windows" && terminal.IsTerminal(int(os.Stdin.Fd())) && r == '\x1a' {
            // ctrl-z detected
            break
        }
    }
}

这里检测程序运行在控制台上,以防止当重定向输入时,输入中包含^Z的情况。\x1a^ZASCII码。

你可能感兴趣的:(go-windows)