最近在学习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
的字符时,程序会正常结束。这里可以看到
^Z
无法关闭标准输入,否则我们应该看到1234
而不是上面的输出^Z
也会被当做一个普通字符^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
是^Z
的ASCII
码。