最近在学习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码。
本文深入探讨了在使用Go语言进行编程时,如何正确处理标准输入,特别是针对Ctrl-Z的使用与检测,避免常见问题。通过分析C++程序示例,解释了在不同操作系统下Ctrl-Z的作用,并提供了在Windows环境下检测Ctrl-Z的方法。
1万+

被折叠的 条评论
为什么被折叠?



