c++ - 检查错误时重载 >> 运算符中的无限循环

标签 c++ error-handling operator-overloading infinite-loop cin

std::istream & operator >>(std::istream & ins, Rational & target)
{
    int num, den;
    char symb;          
    std::cout << "Please enter a rational number: ";
    ins >> num >> symb >> den;
    std::cout << std::endl;

    if(validateInput(num, symb, den)){
        target = Rational(num, den);            
        return ins;
    }
    else{
        std::cin >> target;
    }
}

bool validateInput(int num, char symb, int den)
{
    if(symb != '/'){
        std::cout << "Error: Illegal format. Please use '2/4'." << std::endl;
        return false;
    }
    if((static_cast<int>(num) != num) && (static_cast<int>(den) != den)){
        std::cout << "Error: Not a valid rational number." << std::endl;
        return false;
    }
    if(den == 0){
        std::cout << "Error: Cannot divide by 0." << std::endl;
        return false;
    }

    return true;
}

它接受格式为“x/y”的有理数,例如 2/4。 如果我输入正确,它工作正常。如果我输入 2p4,它会给出正确的错误(我遗漏了一个“/”),然后要求输入一个新号码。如果分母是0,也会报错,要求重新取数。

但检查它是否为有效数字似乎不起作用。如果我输入“a/4”,它将无限循环直到崩溃。我不知道为什么。检查调试器,它返回到 ins >> 语句,但不要求用户提供任何信息。

我假设我的逻辑在某处是错误的。请注意,我对 C++ 还很陌生,还在学习中。我之前尝试过异常处理,但仍然没有正确学习,所以我又选择了我更熟悉的东西。

谢谢!

最佳答案

问题的基本要点是,如果流的状态变坏,C++ 流格式化提取运算符将停止工作,您必须重置状态才能使它们再次工作。

您还有其他问题。

首先,您的验证功能表明您缺乏经验:static_cast<int>(intval) == intval将永远是真实的并且不会证实任何事情。其次,您无法验证您确实已经成功地从流中提取值(这是无限循环的原因:您所做的就是一遍又一遍地验证失败。)

因此,当您提取值时,您应该验证一切正常,如下所示:

int num, den;
char symb;
// Remember to flush unfinished lines
std::cout << "Please enter a rational number: " << std::flush; 
if (std::cin >> num >> symb >> den)
    // you extracted an integer, a character and an integer succesfully
    // perhaps check that the character is '/' and denominator is non-zero
else
    // there was an error: what should we do?

“我们应该做什么”部分远非显而易见:您可以只是重置流并从中删除第一个有问题的字节,然后重试,如果您认为这是合理的(并且直观的) .然而,由于数字太大,提取也可能失败,在这种情况下,这可能会导致奇怪的行为:考虑以下输入(在具有常见 long 大小的实现上):

3111111111111111111111111111111111111111111111111/3

如果您没有任何实际规范,需要考虑的是面向行的输入:首先读取一行并尝试解析它;如果它看起来不正常,请忽略它并尝试下一个。

关于c++ - 检查错误时重载 >> 运算符中的无限循环,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/12224414/

相关文章:

C++析构函数删除共享内存

c++ - std::getline 和 posix 信号

C++从ifstream中拆分字符串并将它们放在单独的数组中

ruby-on-rails - 使用redirect_to后如何停止 Controller 执行? (使用 rails )

java - 了解 try & catch 以及错误处理

android - Android中错误处理的正确方法

c# - 用于序列化 C++ 和反序列化为 C# 类的 Protocol Buffer

c++ - 静态库与动态库

c++ - 为什么在重载下标运算符时使用 const int 参数?

C++:如果替换 typedef,可以解决不明确的重载