161 / 153 / 92
Регистрация: 18.11.2015
Сообщений: 677
1

Некорректная работа cin.fail() при вводе смешанных данных

30.01.2016, 21:52. Показов 3721. Ответов 8
Метки нет (Все метки)

Дарова, у меня есть код, где присутствует проверка на ввод неправильных данных(в моем случае всяких букв), но работает она не так, как мне бы хотелось. У меня пользователь должен ввести стоимость детали, и у меня, как я уже сказал, есть проверка на ввод неправильных данных (букв), но дело в том, что если я ввожу СНАЧАЛА цифры какие-то, а ПОТОМ буквы, то программа думает, что ошибок нету и пропускает дальше. Так вот, как мне сделать так, чтобы программа искала в вводе данных любой намек на буквы и не пропускала дальше?
Вот код:
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
#include <iostream>
#include <string>
#include <limits>
#include <conio.h>
 
struct part {
    std::string name;
    double cost;
};
 
int main() {
 
    const int size = 4;
 
    part apart[size];
 
    std::cout << "===============================================================================" << "\n\n";
    for (int i = 0; i < size; i++) {
        std::cout << "Enter the name of part number " << i + 1 << ": ";
        std::getline(std::cin, apart[i].name);
    enterConst:
        std::cout << "Enter the price of '" << apart[i].name << "' $: ";
        std::cin >> apart[i].cost;
        if (std::cin.fail()) {
            std::cin.clear();
            std::cin.ignore(std::numeric_limits<std::streamsize>::max(), '\n');
            goto enterConst;
        }
        std::cin.ignore(std::numeric_limits<std::streamsize>::max(), '\n');
    }
    std::cout << "\n\n===============================================================================" << "\n\n";
 
    for (int i = 0; i < size; i++) {
        std::cout << "Part number " << i + 1 << ":\n";
        std::cout << "Name: " << apart[i].name << std::endl;
        std::cout << "Cost: $ " << apart[i].cost << std::endl;
        std::cout << "\n===============================================================================" << "\n";
    }
 
    _getch();
}
0
Лучшие ответы (1)
Programming
Эксперт
94731 / 64177 / 26122
Регистрация: 12.04.2006
Сообщений: 116,782
30.01.2016, 21:52
Ответы с готовыми решениями:

Защита от дурака при вводе текста с помощью: cin.get cin.clear cin.sync
Доброго времени суток. На С++ учусь с недавних пор. Имеется стандартная &quot;защита от дурака&quot; на ввод....

Некорректная работа программы,при вводе отрицательного числа
Здравствуйте.Есть некий // { int c; char array; printf( &quot;Vvod: &quot; ); ...

Пробел при вводе данных c применением cin
Допустим имеется следующая запись cin&gt;&gt;n1&gt;&gt;n2&gt;&gt;n3; Спрашивается как оператор разграничивает...

При вводе данных программа пропускает операции cin
Описать структуру с именем note, содержащую следующие поля:  фамилия, имя;  номер...

8
161 / 153 / 92
Регистрация: 18.11.2015
Сообщений: 677
30.01.2016, 21:56  [ТС] 2
Вот пример:
Некорректная работа cin.fail() при вводе смешанных данных
0
nd2
3433 / 2812 / 1249
Регистрация: 29.01.2016
Сообщений: 9,426
30.01.2016, 22:23 3
Цитата Сообщение от meJevin Посмотреть сообщение
Так вот, как мне сделать так, чтобы программа искала в вводе данных любой намек на буквы и не пропускала дальше?
C++
1
2
3
4
5
6
7
8
// в строке ввода только цифры, перед которыми могут быть + или -.
int a;
while (!(cin >> a) || (cin.peek() != '\n'))
{
    cin.clear();
    cin.ignore(numeric_limits<streamsize>::max(), '\n');
    cout << "Error!" << endl;
}
0
161 / 153 / 92
Регистрация: 18.11.2015
Сообщений: 677
30.01.2016, 22:25  [ТС] 4
Цитата Сообщение от nd2 Посмотреть сообщение
int a;
Куда это вставить
0
nd2
3433 / 2812 / 1249
Регистрация: 29.01.2016
Сообщений: 9,426
30.01.2016, 22:27 5
Цитата Сообщение от meJevin Посмотреть сообщение
Куда это вставить
Для твоего кода так пробуй:
C++
1
2
3
4
5
6
7
8
enterConst:
        std::cout << "Enter the price of '" << apart[i].name << "' $: ";
        
        if (!(std::cin >> apart[i].cost) || (std::cin.peek() != '\n')) {
            std::cin.clear();
            std::cin.ignore(std::numeric_limits<std::streamsize>::max(), '\n');
            goto enterConst;
        }
0
161 / 153 / 92
Регистрация: 18.11.2015
Сообщений: 677
30.01.2016, 22:38  [ТС] 6
Цитата Сообщение от nd2 Посмотреть сообщение
Для твоего кода так пробуй:
Спасибо огромное)

Добавлено через 8 минут
Цитата Сообщение от nd2 Посмотреть сообщение
Для твоего кода так пробуй:
О, а пока ты тут. Не скажешь, как работает cin.peek(), если не трудно? Потому что вся документация какая-то странная, без cin.peek() он будет пропускать, если я пробел поставлю, но почему? Он же следующий символ проверяет, или нет?
0
nd2
3433 / 2812 / 1249
Регистрация: 29.01.2016
Сообщений: 9,426
30.01.2016, 23:02 7
Лучший ответ Сообщение было отмечено meJevin как решение

Решение

Цитата Сообщение от meJevin Посмотреть сообщение
Не скажешь, как работает cin.peek(), если не трудно? Потому что вся документация какая-то странная, без cin.peek() он будет пропускать, если я пробел поставлю, но почему? Он же следующий символ проверяет, или нет?
Вопрос не очень понял. Да, проверяет следующий символ. Если ввод правильный (до '\n', или, как в твоём примере на скрине, до первой не цифры), но в строке ввода дальше не '\n' (т.е., до '\n' есть ещё не цифры), то будет считаться ошибкой.

Добавлено через 3 минуты
Какие не цифры до '\n' - это не важно. Пробел, или любая другая не цифра - всё равно будет считаться ошибкой.

Добавлено через 4 минуты
Или тебе нужно, чтобы если только пробелы, в строке ввода, после числа, то не считалось за ошибку?
0
161 / 153 / 92
Регистрация: 18.11.2015
Сообщений: 677
30.01.2016, 23:32  [ТС] 8
Цитата Сообщение от nd2 Посмотреть сообщение
то не считалось за ошибку
Да не, все норм) Спасибо за объяснение
0
nd2
3433 / 2812 / 1249
Регистрация: 29.01.2016
Сообщений: 9,426
30.01.2016, 23:50 9
Вот вариант, если впереди и позади числа только пробелы, то будет считаться, что нет ошибки:
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
enterConst:
    std::cout << "Enter the price of '" << apart[i].name << "' $: ";
    if ((!(std::cin >> apart[i].cost) || (std::cin.peek() != '\n')))
    {
        bool flag = true;
        if (!std::cin)
        {
            std::cin.clear();
            std::cin.ignore(std::numeric_limits<std::streamsize>::max(), '\n');
            flag = false;
        }
        else
        {
            char ch;
            while ((ch = std::cin.get()) != '\n')
                if (ch != ' ')
                    flag = false;
        }
        if (!flag)
            goto enterConst;
    }
0
IT_Exp
Эксперт
87844 / 49110 / 22898
Регистрация: 17.06.2006
Сообщений: 92,604
30.01.2016, 23:50
Помогаю со студенческими работами здесь

Поведение cin при вводе некорректного типа данных
Был создан цикл: while (err) //bool err = true { try { cin &gt;&gt; arrsz // int...

Зацикливание приложения при вводе в cin данных с пробелом
Есть такой код ( курсовой проект, не суть). Проблема такова: при вводе в переменную vak значения...

Некорректная работа cin.getline()
Доброго времени суток,господа знатоки. у меня есть двумерный массив char куда мне нужно записать...

Некорректная работа cin.fileget()
Здорова господа!!! Пытаюсь разобраться с потоками ввода/вывода вот есть код: #include...

Отбрасывание "мусорных" данных при вводе числа через cin
Имеется задача, когда необходимо ввести с клавиатуры число и обработать его. Если делать...

cin.eof и cin.fail
Для чего они нужны и какая разница между ними; cin.eof и cin.fail Я так понимаю они(оба)...


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
9
Ответ Создать тему
Опции темы

КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2023, CyberForum.ru