Проверка на допустимый целочисленный ввод в C++

Так что я видел, что многие люди спрашивают об этом, а не много твердых ответов, плавающих в сети. большинство из них просто проверяют, что вместо строки было помещено целое число, но если было введено число с плавающей запятой, оно усекает нижнюю половину или, если целые числа и символы являются целыми, оно усекает символы. Мне нужна помощь в написании фрагмента кода, который проверяет ввод пользователя и просит пользователя повторить попытку, если его ввод недопустим или комбинация действительный / недействительный. я думаю, что основная идея заключалась в том, чтобы создать строку, чтобы она принимала что-либо, затем использовать sstream для манипулирования, а затем вернуться к int, если ввод был допустимым, но мне не удалось проверить другие части. если кто-то сталкивается с этим или может помочь мне, пожалуйста, свяжите меня с этим. я опубликую свой код, когда получу хорошее представление о том, что делать.

4 ответа

Решение

Предполагая, что вы не можете использовать boost::lexical_castВы можете написать свою собственную версию:

#include <sstream>
#include <iostream>
#include <stdexcept>
#include <cstdlib>
template <class T1, class T2>
T1 lexical_cast(const T2& t2)
{
  std::stringstream s;
  s << t2;
  T1 t1;
  if(s >> std::noskipws >> t1 && s.eof()) {
    // it worked, return result
    return t1;
  } else {
    // It failed, do something else:
    // maybe throw an exception:
    throw std::runtime_error("bad conversion");
    // maybe return zero:
    return T1();
    // maybe do something drastic:
    exit(1);
  }
}



int main() {
  std::string smin, smax;
  int imin, imax;

  while(std::cout << "Enter min and max: " && std::cin >> smin >> smax) {
    try {
      imin = lexical_cast<int>(smin);
      imax = lexical_cast<int>(smax);
      break;
    } catch(std::runtime_error&) {
      std::cout << "Try again: ";
      continue;
    }
  }

  if(std::cin) {
    std::cout << "Thanks!\n";
  } else {
    std::cout << "Sorry. Goodbye\n";
    exit(1);
  }
}

Вы можете использовать функции преобразования строк в C++11, такие как stol

try
{
    std::string value = ...;
    long number = std::stol(value); 
}
catch (std::invalid_argument const& e)
{
    // no conversion could be performed
}

Обновление после комментариев: Visual C++ 11 поставляется с реализациями Visual Studio 2012 std::stol в качестве удобной обертки вокруг strtol объявлено в <cstdlib>, Я думаю, что можно с уверенностью предположить, что большинство реализаций C++11 определяют его наиболее оптимальным способом, не достигая std::stringstream техника.

Функция C strtol (и это братья и сестры) сможет сказать вам, если строка, поданная к нему, полностью израсходована.

 std::string str;
 char *endptr;
 std::cin >> str;
 long x = std::strtol(str.c_str(), &endptr, 0); 
 if (*endptr != 0) 
    cout << "That's not a valid number...";

Я не знаю, есть ли какие-нибудь классы в стандартной C++ lib, которые инкапсулируют примитивные типы, как в java, но вот как будет выглядеть простая и очень базовая реализация

class Integer {
   private:
       int value;
       void parse(string);
   public:
       Integer(string);
       int intValue();
};

Integer::Integer(string sint) { parse(sint); }
int Integer::intValue() { return value; }

void Integer::parse(string sint) {
    string::iterator its = sint.begin();
    while(its != sint.end() && (! (*its < '0' || *its > '9'))) {
        its++;
    }
    if(its != sint.end()) {
        throw sint + ": Input is not a valid integer.";
    }
    value = atoi(sint.c_str());
}
Другие вопросы по тегам