繁体   English   中英

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

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

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

#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;
    }
}

我发现已经成功读取了年龄,但没有读取名称。 这是输入和 output:

 Input: "10" "Mr. Whiskers" Output: "My cat is 10 years old and their name is "

为什么 output 中省略了这个名字? 我已经给出了正确的输入,但代码以某种方式忽略了它。 为什么会这样?

为什么会这样?

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

 "10\n"

当您从终端提交时选择EnterReturn时,总是会在您的输入中附加一个换行符。 它也用于文件中以移动到下一行。 在提取到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\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 until 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 their name is " << name << std::endl;
    
}

额外的:

如果用户输入a \n10\n \nmr. whiskey 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 untill input is `int or float`
{
    cin>> *pInput;
    /*-----------check input validation----------------*/
    while (!cin) 
    {
        cin.clear();
        cin.ignore(100, '\n');
        cout<<"Invalid Input Type.\nEnter again: ";
        cin >>*pInput;
    }
    /*-----------checked input validation-------------*/
}

我真的很想知道。 C++ 有一个专门的函数来吃掉任何剩余的或任何空白。 它被称为std::ws 然后,您可以简单地使用

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

这应该是惯用的方法。 对于应使用的格式化输入到未格式化输入之间的每次转换。

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

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

请阅读这里

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM