我有以下代码提示用户输入猫的年龄和名字:
#include <iostream>
#include <string>
int main()
{
int age;
std::string name;
std::cin >> age;
std::getline(std::cin, name);
if (std::cin)
{
std::cout << "My cat is " << age << " years old and their name is " << name << std::endl;
}
}
我发现是成功读取了年龄,但是没有读取名字。这是输入和输出:
Input: "10" "Mr. Whiskers" Output: "My cat is 10 years old and their name is "
为什么输出中省略了名称?我已经给出了正确的输入,但代码以某种方式忽略了它。为什么会这样?
这与您自己提供的输入无关,而是与
std::getline()
具有的默认行为有关。当您输入年龄 (std::cin >> age
) 时,您不仅提交了以下字符,而且当您键入 Enter: 时,还会在流中附加一个隐式换行符
"10\n"
当您从终端提交时选择 Enter 或 Return 时,总是会在输入后附加一个换行符。它也用于文件中以移动到下一行。提取到
age
后,换行符留在缓冲区中,直到下一个 I/O 操作被丢弃或读取。当控制流到达std::getline()
时,会看到"\nMr. Whiskers"
,开头的换行符会被丢弃,但输入操作会立即停止。发生这种情况的原因是因为 std::getline()
的工作是尝试读取字符并在找到换行符时停止。因此,您输入的其余部分将留在未读的缓冲区中。
cin.ignore()
要解决此问题,一种选择是在执行
std::getline()
之前跳过换行符。您可以通过在第一次输入操作后调用 std::cin.ignore()
来完成此操作。它会丢弃下一个字符(换行符),这样它就不再碍事了。
std::cin >> age;
std::cin.ignore();
std::getline(std::cin, name);
assert(std::cin);
// Success!
std::ws
另一种丢弃空格的方法是使用
std::ws
函数,它是一个操纵器,旨在从输入流的开头提取和丢弃前导空格:
std::cin >> age;
std::getline(std::cin >> std::ws, name);
assert(std::cin);
// Success!
std::cin >> std::ws
表达式在 std::getline()
调用之前(以及 std::cin >> age
调用之后)执行,以便删除换行符。
不同之处在于,
ignore()
仅丢弃 1 个字符(或给定参数时 N 个字符),而 std::ws
继续忽略空白,直到找到非空白字符。因此,如果您不知道下一个标记之前有多少空格,您应该考虑使用它。
当你遇到这样的问题时,通常是因为你将格式化输入操作与非格式化输入操作结合在一起。格式化输入操作是指您获取输入并将其格式化为某种类型。这就是
operator>>()
的用途。无格式输入操作是除此之外的任何东西,如 std::getline()
、std::cin.read()
、std::cin.get()
等。这些函数不关心输入的格式,只处理原始文本。
如果你坚持使用单一类型的格式,那么你可以避免这个恼人的问题:
// Unformatted I/O
std::string age, name;
std::getline(std::cin, age);
std::getline(std::cin, name);
或
// Formatted I/O
int age;
std::string firstName, lastName;
std::cin >> age >> firstName >> lastName;
如果您选择使用无格式操作将所有内容读取为字符串,您可以在之后将它们转换为适当的类型。
如果您按以下方式更改初始代码,一切都会好的:
if ((cin >> name).get() && std::getline(cin, state))
发生这种情况是因为隐式换行符也称为换行符
\n
被附加到来自终端的所有用户输入,因为它告诉流开始新行。在检查多行用户输入时,您可以使用 std::getline
安全地考虑到这一点。 std::getline
的默认行为将从输入流对象读取所有内容,包括换行符 \n
,在本例中为 std::cin
。
#include <iostream>
#include <string>
int main()
{
std::string name;
std::string state;
if (std::getline(std::cin, name) && std::getline(std::cin, state))
{
std::cout << "Your name is " << name << " and you live in " << state;
}
return 0;
}
Input: "John" "New Hampshire" Output: "Your name is John and you live in New Hampshire"
既然上面的人都已经回答了输入
10\nMr Whisker\n
的问题,我想换一种方式来回答。上面的所有解决方案都发布了缓冲区是否像 10\nMr Whisker\n
的代码。但是,如果我们不知道用户在提供输入时会如何表现怎么办。用户可能会错误地键入 10\n\nMr. Whisker\n
或 10 \n\n Mr. whisker\n
。在这种情况下,上面的代码可能不起作用。所以,我使用下面的函数来获取字符串输入来解决这个问题。
string StringInput() //returns null-terminated string
{
string input;
getline(cin, input);
while(input.length()==0)//keep taking input as long as valid string is taken
{
getline(cin, input);
}
return input.c_str();
}
所以,答案是:
#include <iostream>
#include <string>
int main()
{
int age;
std::string name;
std::cin >> age;
name = StringInput();
std::cout << "My cat is " << age << " years old and it's name is " << name << std::endl;
}
额外:
如果用户输入
a \n10\n \nmr. whiskey
;
要检查 int
输入是否有效,此函数可用于检查 int
输入(如果 char
作为输入而不是 int
,程序将有未定义的行为):
//instead of "std::cin>>age;" use "get_untill_int(&age);" in main function.
void get_Untill_Int(int* pInput)//keep taking input until input is `int or float`
{
cin>> *pInput;
/*-----------check input validity----------------*/
while (!cin)
{
cin.clear();
cin.ignore(100, '\n');
cout<<"Invalid Input Type.\nEnter again: ";
cin >>*pInput;
}
/*-----------checked input validity-------------*/
}