区分int和double

时间:2015-04-03 06:27:44

标签: c++ int double invalid-characters

我搜索过这个答案,似乎没有人知道如何解决这个错误。我希望输入严格地为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);

它似乎几乎可以工作。它发送错误,但只有在进入下一个循环后才会发送。我怎样才能解决这个问题?为什么它仍然显示错误信息但仍然在显示之前继续前进?

4 个答案:

答案 0 :(得分:2)

输入可以是整数或浮点数的表示以外的其他内容。

请记住,数字不是它们的表示形式:9(十进制),017(八进制,àlaC),0b1001(二进制,àlaOcaml),{ {1}}(罗马符号),IX(算术表达式),8+1(法语)都是相同数字9的表示。

因此,您必须决定是否接受neuf9 x 9(数字后面有几个空格)等输入,...更一般地说,您必须定义什么是可接受的输入(如果输入是否在行尾结束,如果应该接受空格或标点,等等......)。

您可以阅读整行(例如使用std::getline)并使用例如sscanf(其中控件格式很有用,%n)或std::stol(使用结束指针)解析它的项目计数也是如此< / p>

另请注意,您的问题(“区分int和double”)的措词是错误的。 C ++中没有单个“sscanfint”类型(但double是标量类型,而int是C ++中的标量类型,您可以定义一个doubletagged 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;
}