为什么 std::getline() 在格式化提取后跳过输入?

Why does std::getline() skip input after a formatted extraction?

提问人:David G 提问时间:2/5/2014 最后编辑:Jan SchultkeDavid G 更新时间:9/27/2023 访问量:33745

问:

我有以下一段代码,提示用户输入他们的猫的年龄和名字:

#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 "

为什么输出中省略了该名称?我已经给出了正确的输入,但代码不知何故忽略了它。为什么会这样?

C++ 输入 IOSTREAM getline istream

评论

1赞 jww 11/12/2018
我相信也应该按预期工作。(除了下面的答案)。std::cin >> name && std::cin >> std::skipws && std::getline(std::cin, state)

答:

178赞 David G 2/5/2014 #1

为什么会这样?

这与您自己提供的输入关系不大,而是与默认行为有关。当您提供 age () 的输入时,您不仅提交了以下字符,而且在键入时,还会将隐式换行符附加到流中:std::getline()std::cin >> ageEnter

"10\n"

当您选择或从终端提交时,始终会将换行符附加到您的输入中。它还用于文件中,以便移动到下一行。换行符在提取后保留在缓冲区中,直到下一个 I/O 操作被丢弃或读取。当控制流到达 时,它将看到并且开头的换行符将被丢弃,但输入操作将立即停止。发生这种情况的原因是 的工作是尝试读取字符并在找到换行符时停止。因此,其余的输入在缓冲区中处于未读状态。EnterReturnagestd::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::wsstd::getline()std::cin >> age

区别在于,它只丢弃 1 个字符(或给定参数时丢弃 N 个字符),并继续忽略空格,直到找到非空格字符。因此,如果您不知道下一个令牌之前会有多少空格,您应该考虑使用它。ignore()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;

如果选择使用未格式化的操作将所有内容读取为字符串,则可以在之后将它们转换为适当的类型。

评论

1赞 Fred Larson 8/20/2016
为什么不干脆呢?if (getline(std::cin, name) && getline(std::cin, state))
0赞 David G 8/20/2016
@FredLarson 好点子。尽管如果第一次提取是整数或任何不是字符串的内容,它将不起作用。
1赞 Fred Larson 8/20/2016
当然,这里的情况并非如此,以两种不同的方式做同样的事情是没有意义的。对于整数,您可以将该行放入字符串中,然后使用 ,但就不那么清楚了。但我倾向于只用于面向行的输入,然后以任何有意义的方式处理解析行。我认为它不太容易出错。std::stoi()std::getline()
0赞 David G 8/20/2016
@FredLarson同意了。如果我有时间,也许我会把它加进去。
2赞 David G 4/4/2020
@Albin 您可能想要使用的原因是,如果要捕获给定分隔符之前的所有字符并将其输入到字符串中,默认情况下,该字符串是换行符。如果这些数量的字符串只是单个单词/标记,则可以使用 轻松完成此工作。否则,您将第一个数字输入到整数中,并在下一行调用,然后在使用 .std::getline()X>>>>cin.ignore()getline()
14赞 Boris 3/26/2014 #2

如果按以下方式更改初始代码,则一切都会好起来的:

if ((cin >> name).get() && std::getline(cin, state))

评论

3赞 David G 3/26/2014
谢谢。这也将起作用,因为会消耗下一个字符。还有我之前在回答中提出的建议。get()(std::cin >> name).ignore()
0赞 Boris 3/26/2014
"..工作,因为 get()...”是的,没错。很抱歉在没有细节的情况下给出答案。
4赞 Fred Larson 8/20/2016
为什么不干脆呢?if (getline(std::cin, name) && getline(std::cin, state))
4赞 Justin Randall 2/21/2018 #3

发生这种情况是因为隐式换行符(也称为换行符)被附加到来自终端的所有用户输入中,因为它告诉流开始新行。在检查多行用户输入时,可以使用 std::getline 安全地考虑这一点。的默认行为是,将从输入流对象(在本例中)读取所有内容,包括换行符。\nstd::getline\nstd::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"
2赞 Miraz 6/13/2021 #4

由于上面的每个人都回答了输入问题,我想回答不同的方法。上面的所有解决方案都发布了如果缓冲区像 .但是,如果我们不知道用户在提供输入时会如何表现怎么办?用户可能会键入或错误地键入。在这种情况下,上述代码可能不起作用。因此,我使用下面的函数来获取字符串输入来解决问题。10\nMr Whisker\n10\nMr Whisker\n10\n\nMr. Whisker\n10 \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. whiskeyintintcharint


//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-------------*/
}
2赞 A M 11/19/2021 #5

我真的很纳闷。C++ 有一个专用函数,用于吃掉任何剩余或任何空格。它被称为 std::ws。然后,您可以简单地使用

std::getline(std::cin >> std::ws, name);

这应该是偶像主义的方法。对于应使用的格式化输入和未格式化输入之间的每次转换。

如果我们不是在谈论空格,而是在需要数字的地方输入例如字母,那么我们应该遵循 CPP 引用并使用

.ignore(std::numeric_limits<std::streamsize>::max(), '\n');消除错误的东西。

在这里阅读