C++primer 中引言的程序如下所示,期望输出一行整型例如:3 4 5 6之后回车键能够得到sum值。
但是运行过程中并不能跳出输入的循环,但是输入非整型情况下可以跳出循环。
#include <iostream>
using namespace std;
int main()
{
int number = 0, sum = 0;
while (cin >> number)
{
sum += number;
}
cout << "sum is :" << sum << endl;
return 0;
}
原因分析:(转自https://blog.csdn.net/misayaaaaa/article/details/53332219)
在 Unix 系统中的实际的现象为:
* 当输入为“字符串,回车,Ctrl+D”的时候,跳出循环;
* 当输入为“字符串,Ctrl+D,回车”的时候,不会跳出循环;
* 当输入为“字符串,Ctrl+D,回车,Ctrl+D”的时候,跳出循环;
* 当输入为“字符串,Ctrl+D,Ctrl+D”的时候,跳出循环;
在 Windows 系统中的实际现象待考察。
具体原因解释如下:
输入(cin)缓冲是行缓冲。当从键盘上输入一串字符并按回车后,这些字符会首先被送到输入缓冲区中存储。每当按下回车键后,cin 就会检测输入缓冲区中是否有了可读的数据。cin 还会对键盘上是否有作为流结束标志的 Ctrl+Z 或者 Ctrl+D 键按下作出检查,其检查的方式有两种:阻塞式以及非阻塞式。
阻塞式检查方式指的是只有在回车键按下之后才对此前是否有 Ctrl+Z 组合键按下进行检查,非阻塞式样指的是按下 Ctrl+D 之后立即响应的方式。如果在按 Ctrl+D 之前已经从键盘输入了字符,则 Ctrl+D的作用就相当于回车,即把这些字符送到输入缓冲区供读取使用,此时Ctrl+D不再起流结束符的作用。如果按 Ctrl+D 之前没有任何键盘输入,则 Ctrl+D 就是流结束的信号。
阻塞式的方式有一个特点:只有按下回车之后才有可能检测在此之前是否有Ctrl+Z按下。还有一个特点,如果输入缓冲区中有可读的数据则不会检测 Ctrl+Z(因为有要读的数据,还不能认为到了流的末尾)。还有一点需要知道,Ctrl+Z产生的不是一个普通的ASCII码值,也就是说它产生的不是一个字符,所以不会跟其它从键盘上输入的字符一样能够存放在输入缓冲区。
解决方法一: 在键入number结束的时候采用Fn+F6按键后再按回车键就可以结束;
解决方法二:使用getline函数:
#include <iostream>
#include <string>
#include <sstream>
using namespace std;
int main()
{
string number;
int sum = 0;
while (getline(cin, number))
{
stringstream ss(number);
int x;
while (ss >> x)
sum += x;
cout << sum << endl;
}
cout << "Hello World!\n";
return 0;
}