Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

My cin is being ignored inside a while loop

I am trying to code a simple question and number checker into my first C++ program. Problem is, when I type a string like one two, or three, the program becomes and infinite loop and it ignores the cin function to re-assign lives to a number.

cout << "How many lives would you like 1 (hard), 2 (medium), or 3 (easy)?" << endl;
cin >> lives;


while(lives != 1 && lives != 2 && lives != 3 && !isdigit(lives))
{
    cout << "You need to input a number, not words." << endl;
    cout << "How many lives would you like 1 (hard), 2 (medium), or 3 (easy)?" << endl;
    cin >> lives;
}

Here is my current code with your suggestions:

    cout << "How many lives would you like 1 (hard), 2 (medium), or 3 (easy)?" << endl;
std::cin.ignore();
std::cin.clear();
if (std::cin >> lives)
{


    while(lives != 1 && lives != 2 && lives != 3)
    {
        cout << "You need to input a number, not words." << endl;
        cout << "How many lives would you like 1 (hard), 2 (medium), or 3 (easy)?" << endl;
        cin >> lives;
    }

}
like image 997
Logan Saso Avatar asked Aug 03 '13 19:08

Logan Saso


2 Answers

#include <iostream>
#include <limits>

int main()
{
    int lives = 0;
    std::cout << "How many lives would you like 1 (hard), 2 (medium), or 3 (easy)?" << std::endl;


    while(!(std::cin >> lives) || lives < 1 || lives > 3)
    {
        std::cout << "You need to input a number, not words." << std::endl;
        std::cout << "How many lives would you like 1 (hard), 2 (medium), or 3 (easy)?" << std::endl;
        std::cin.clear();
        std::cin.ignore(std::numeric_limits<std::streamsize>::max(), '\n');
    }

    return 0;
}

Alright. std::cin.clear(); takes care of resetting the fail bits. std::cin.ignore removes any wrong input left in the stream. And I've adjusted the stop condition. (isDigit was a redundant check, if lives is between 1 and 3 then clearly it's a digit).

like image 56
Borgleader Avatar answered Sep 20 '22 11:09

Borgleader


When std::istream can't read a value, it goes into failure mode, i.e., std::failbit is set and the stream yields false when tested. You always want to test if a read operation was successful:

if (std::cin >> value) {
    ...
}

To restore the stream to good state you'd use std::cin.clear() and you probably need to ignore bad characters, e.g., using std::cin.ignore().

like image 26
Dietmar Kühl Avatar answered Sep 18 '22 11:09

Dietmar Kühl