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

标签 c++ input iostream istream c++-faq

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

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

当您从终端提交时选择 EnterReturn 时,总是会在输入后附加一个换行符。它也用于文件中以移动到下一行。提取到 age 后,换行符将留在缓冲区中,直到下一个 I/O 操作被丢弃或读取。当控制流到达std::getline()时,会看到"\nWhiskers",开头的换行符会被丢弃,但是输入操作会立即停止。发生这种情况的原因是因为 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;

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

关于c++ - 为什么 std::getline() 在格式化提取后跳过输入?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38297800/

相关文章:

c++ - 从另一个列表 C++ Qt 中的对象的属性创建一个新的值列表

c++ - 为什么给数组中的第一个元素赋值时必须指定索引号?

html - 输入的样式看起来像超链接

c - float 输出不正确

c++ - << 运算符重写为 cout int 和 double 值

c++ "Run-Time Check Failure #2 - Stack around the variable ' board' 已损坏”

html - 不输入除带有数字属性的输入标记中的 e 字符之外的所有字符

c++ - Cpp 98 标准中的 std::cout 格式

c++ - 交替的 cin/cout 很慢?

c++ - Cython/C++ - 共享 PXD 文件中的头文件路径