当我输入一个字母时,cin 输入(输入是一个 int),而不是打印错误一次,它打印正确一次,然后在循环的其余部分打印 i

cin input (input is an int) when I input a letter, instead of printing back incorrect once, it prints correct once then inc for the rest of the loop

本文关键字:输入 打印 一个 一次 然后 余部 循环 cin int 错误      更新时间:2023-10-16

我正在制作一个乘法练习程序。正如我的标题所说,如果我在控制台中输入一个字母而不是一个数字,它会在第一个字母上说正确,但在其余部分不正确。即使您没有触摸键盘,它仍然会吐出不正确的信息。

ans = table * i;
std::cout << table << " * " << i << " =" << std::endl;
std::cin >> input;
if(input == ans)
{
std::cout << "Correct! " << ans << std::endl;
}
else
{
std::cout << "Incorrect, the answer was " << ans << std::endl;
input = 0;
ans = 0;
}
}

希望这能让您了解代码中发生的情况。这是输出。

Enter the table you'd like to learn.
5
Enter the amount of multiples
3
5 * 0 =
s
Correct! 0
5 * 1 =
Incorrect, the answer was 5
5 * 2 =
Incorrect, the answer was 10
5 * 3 =
Incorrect, the answer was 15

我该怎么做才能解决这个问题?感谢您的输入

有许多方法可以构建各种测试,但是当使用cin(或通常使用任何输入函数(进行输入时,必须考虑输入缓冲区中未读取的任何字符。对于cin,您必须验证三个条件:

  1. .eof()(eofbit( 已设置。要么到达输入结束,要么用户通过按 Ctrl+d手动生成EOF(或在 windoze 上按 Ctrl+z,但请参阅:CTRL+Z 在 Windows 10 中不会生成 EOF(;

  2. .bad()(badbit( 已设置。发生不可恢复的流错误;和

  3. .fail()(failbit( 已设置。发生匹配或其他可恢复故障。发生故障时,输入停止,并且不会从输入缓冲区中提取更多字符

(您可以将 1 和 2 合并为一个测试,因为此时输入已结束(

对于failbit,您必须做两件事。(1( 必须通过调用cin.clear()来清除流错误状态,以及 (2( 必须处理输入缓冲区中未读取的任何字符。通常,这是通过包含<limits>和调用来处理的:

std::cin.ignore(std::numeric_limits<std::streamsize>::max(), 'n');

(这将从输入缓冲区中清除最多INT_MAX个字符,在遇到分隔符(此处'n'(时停止(

在用户输入有效整数之前循环的简短示例可能是:

#include <iostream>
#include <limits>
using namespace std;
int main (void) {
int x = 0;
while (1)       /* loop continually until valid input received */
{
cout << "nenter an integer: ";
if (! (cin >> x) ) {            /* check stream state */
/* if eof() or bad() break read loop */
if (cin.eof() || cin.bad()) {
cerr << "(user canceled or unreconverable error)n";
return 1;
}
else if (cin.fail()) {      /* if failbit */
cerr << "error: invalid input.n";
cin.clear();            /* clear failbit */
/* extract any characters that remain unread */
cin.ignore(numeric_limits<streamsize>::max(), 'n');
}
}
else {  /* on succesful read of int */
/* extract any characters that remain unread */
cin.ignore(numeric_limits<streamsize>::max(), 'n');
break;  /* then break read loop */
}
}
cout << "integer: " << x << 'n';
}

(如开头所述,只要涵盖所有三个条件,您就可以以多种不同的方式构建测试。

此外,您可以通过调用rdstate()来显式检查流位,而不是使用.fail()等进行测试,例如

if (std::cin.rdstate() == std::ios_base::failbit)

示例使用/输出

$ ./bin/cin_validate_int
enter an integer: no
error: invalid input.
enter an integer: "an integer"
error: invalid input.
enter an integer: abc 123
error: invalid input.
enter an integer: 123
integer: 123

仔细查看,如果您有其他问题,请告诉我。

当你执行cin >> input并且它无法将输入解析为数字时,它会设置cinfailbit,当设置failbit时,它实际上不会读取更多的数据,直到你清除它。

您应该通过调用cin.good()来检查错误,并尝试使用cin.clear()清除它们,但您不应该尝试清除某些错误(如 EOF(,而应该退出它们。