我搜索过这个答案,似乎没有人知道如何解决这个错误。我希望输入严格地为int。如果输入是double,我希望它发送错误。
int creatLegs = 0;
string trash;
bool validLegs = true;
do
{
cout << "How many legs should the creature have? ";
cin >> creatLegs;
if(cin.fail())
{
cin.clear();
cin >> trash; //sets to string, so that cin.ignore() ignores the whole string.
cin.ignore(); //only ignores one character
validLegs = false;
}
if (creatLegs > 0)
{
validLegs = true;
}
if (!validLegs)
{
cout << "Invalid value, try again.\n";
}
} while (!validLegs);
它似乎几乎可以工作。它发送错误,但只有在进入下一个循环后才会发送。我怎样才能解决这个问题?为什么它仍然显示错误信息但仍然在显示之前继续前进?
答案 0 :(得分:2)
输入可以是整数或浮点数的表示以外的其他内容。
请记住,数字不是它们的表示形式:9
(十进制),017
(八进制,àlaC),0b1001
(二进制,àlaOcaml),{ {1}}(罗马符号),IX
(算术表达式),8+1
(法语)都是相同数字9的表示。
因此,您必须决定是否接受neuf
或9 x
9
(数字后面有几个空格)等输入,...更一般地说,您必须定义什么是可接受的输入(如果输入是否在行尾结束,如果应该接受空格或标点,等等......)。
您可以阅读整行(例如使用std::getline)并使用例如sscanf
(其中控件格式很有用,
%n
)或std::stol(使用结束指针)解析它的项目计数也是如此< / p>
另请注意,您的问题(“区分int和double”)的措词是错误的。 C ++中没有单个“sscanf
或int
”类型(但double
是标量类型,而int
是C ++中的标量类型,您可以定义一个double
,tagged union可以容纳其中任何一个)。 AFAIU,如果您声明class
,则使用int x;
,用户输入std::cin >> x;
点后面的数字12.64
将不会被解析,64
会变成12岁。
答案 1 :(得分:0)
int creatLegs = 0;
do
{
cout << "How many legs should the creature have? ";
cin >> creatLegs; // trying to get integer
if(!cin.fail()) // if cin.fail == false, then we got an int and leave loop
break;
cout << "Invalid value, try again.\n"; // else show err msg and try once more
cin.clear();
} while (1);
答案 2 :(得分:0)
我认为你应该将数据读取为字符串,然后通过char检查它是否为char,以验证它是否为整数 - 如果每个字符都是数字,那么我们有整数,我们可以解析它。
stream的问题是,如果你试图读取整数但是传递了十进制,那么它会读取到点的数字。这部分是一个正确的整数,因此cin.fail()
返回false
。
示例代码:
#include <iostream>
#include <string>
#include <cctype>
#include <cstdlib>
using namespace std;
int main() {
int creatLegs = 0;
bool validLegs = true;
do
{
cout << "How many legs should the creature have? ";
string input;
getline(cin, input);
validLegs = true;
for (string::const_iterator i = input.begin(); validLegs && i != input.end(); ++i) {
if (!isdigit(*i)) {
validLegs = false;
}
}
if (!validLegs)
{
cout << "Invalid value, try again.\n";
} else {
creatLegs = atoi(input.c_str());
}
} while (!validLegs);
cout << creatLegs << endl;
}
这当然不是一个完美的解决方案。如果有任何前导或尾随空格(或任何其他字符,如+
或-
),程序将失败。但是如果需要的话,你总是可以添加一些代码来处理这些情况。
答案 3 :(得分:0)
这个问题已经有了一个公认的答案,但我会提供一个解决方案来处理所有积分的数字,即使那些表示为浮点数(没有小数部分)并且拒绝包含任何内容的输入除了号码后面的空格。
可接受值的示例,这些值均代表数字4:
4 4. 4.0 +4 004.0 400e-2
拒绝值的示例:
3.999999 4.000001 40e-1x 4,
#include <iostream>
#include <sstream>
#include <cctype>
#include <string>
using namespace std;
bool get_int( const string & input, int & i ) {
stringstream ss(input);
double d;
bool isValid = ss >> d;
if (isValid) {
char c;
while( isValid && ss >> c ) isValid = isspace(c);
if (isValid) {
i = static_cast<int>(d);
isValid = (d == static_cast<double>(i));
}
}
return isValid;
}
int main( int argc, char *argv[] )
{
int creatLegs = 0;
bool validLegs = false;
do
{
string line;
do {
cout << "How many legs should the creature have? ";
} while (not getline (cin,line));
validLegs = get_int( line, creatLegs );
if (creatLegs <= 0)
{
validLegs = false;
}
if (not validLegs)
{
cout << "Invalid value, try again." << endl;
}
} while (not validLegs);
cout << "Got legs! (" << creatLegs << ")" << endl;
return 0;
}
如果你想要严格的整数(没有小数周期而没有科学记数法),那么使用这个更简单的 get_int 函数:
bool get_int( const string & input, int & i ) {
stringstream ss(input);
bool isValid = ss >> i;
if (isValid) {
char c;
while(isValid && ss >> c) isValid = isspace(c);
}
return isValid;
}