c++ - 如何成功清除流?

标签 c++ c++11

我正在编写一个程序,该程序需要根据希腊命理学找出丰数、亏数和完全数。这里的一个问题是确保只插入自然数;更具体地说,确保没有插入任何字母。 我尝试检查流是否“损坏”,如果为真,则打印适当的消息。它有效,但只有一次,我不明白为什么。 这是代码。

 #include <iostream>
 int main ()
{
double n(0);
int sum(0);
int i(0);

std::cout << "Enter a natural number or 0 for exit: ";
std::cin >> n;

int n1 = n;
do {
    for(i = 1; i < n1; i++) {
        if(n1 % i == 0) {
            sum+=i;
        }
    }
    if(!std::cin) {
        std::cout << "Not a natural number!" << std::endl;
        std::cout << "Enter a natural number or 0 for exit: ";
    }
    else if(n != int(n)) {
        std::cout << "Not a natural number!" << std::endl;
        std::cout << "Enter a natural number or 0 for exit: ";
    }
    else if(n1 < 0) {
        std::cout << "Not a natural number!" << std::endl;
        std::cout << "Enter a natural number or 0 for exit: ";
    }
    else if(suma < n1) {
        std::cout << "Number " << n << " is deficient!" << std::endl;
        std::cout << "Enter a natural number or 0 for exit: ";
    }
    else if(suma > n1) {
        std::cout << "Number " << n << " is abudant!" << std::endl;
        std::cout << "Enter a natural number or 0 for exit: ";
    }
    else if(suma == n1) {
        std::cout << "Number " << n << " is perfect!" << std::endl;
        std::cout << "Enter a natural number or 0 for exit: ";
    }
    suma = 0;
    std::cin.clear();
    std::cin.ignore(1000, '\n');
    std::cin >> n;
    n1 = int(n);
} while(n != 0);
std::cout << "Goodbye!";

return 0;
}

最佳答案

问题似乎是一个逻辑问题。您提示重新输入 n , 然后

std::cin.clear();
std::cin.ignore(1000, '\n');
std::cin >> n;

这不会“恢复”流,会丢弃 'n' 的用户输入让你陷入困境。看来您需要重新排序您的输入和循环以在每次迭代中处理一次输入。类似于以下内容:

#include <iostream>

int main (void) {

    double n(0);

    do {
        int sum(0);
        int i(0);

        std::cout << "Enter a natural number or 0 for exit: ";
        std::cin >> n;
        int n1 = n;

        for(i = 1; i < n1; i++)
            if(n1 % i == 0)
                sum+=i;

        if(n != int(n))
            std::cout << "Not a natural number!" << std::endl;
        else if(n1 < 0)
            std::cout << "Not a natural number!" << std::endl;
        else if(sum < n1)
            std::cout << "Number " << n << " is deficient!" << std::endl;
        else if(sum > n1)
            std::cout << "Number " << n << " is abudant!" << std::endl;
        else if(sum == n1)
            std::cout << "Number " << n << " is perfect!" << std::endl;

        sum = 0;

    } while(n != 0);

    std::cout << "Goodbye!\n";

    return 0;
}

都不是.ignore.clear是必要的。数字转换和 .clear 忽略前导空格只需设置 'goodbit'在具有清除所有错误标志的效果的流上。 (参见 std::basic_ios::clear )也没有真正需要检查 if (!std::cin) .它没有坏处,它对您的代码来说只是多余的。

示例使用/输出

$ ./bin/inputnumber
Enter a natural number or 0 for exit: 128
Number 128 is deficient!
Enter a natural number or 0 for exit: 30
Number 30 is abudant!
Enter a natural number or 0 for exit: 37
Number 37 is deficient!
Enter a natural number or 0 for exit: 2
Number 2 is deficient!
Enter a natural number or 0 for exit: 1
Number 1 is deficient!
Enter a natural number or 0 for exit: 10
Number 10 is deficient!
Enter a natural number or 0 for exit: 12
Number 12 is abudant!
Enter a natural number or 0 for exit: 0
Number 0 is perfect!
Goodbye!

我不是 100% 清楚你的目标,所以如果我误解了,请发表评论,我很乐意提供进一步的帮助。


让您的输入丢弃非数字

Rose,经过评论中的讨论,您的问题似乎与 n1 有关正在分配0如果出现输入错误(如猫踩键盘导致输入 ";sdkfgj")。这也可以处理,而且您在查看两个.clear 时都走在正确的轨道上和 .ignore .至此,将它们组合在一起的方式有一点逻辑。

当遇到“如何确保用户只输入'X'?”的问题时,通常的做法是不断循环,直到用户输入满足您设置的接受输入的条件。在这里,您需要输入一个数值。所以方法是循环,提示输入,然后在用户输入值后检查流状态。如果流出错,需要.clear()错误,然后 .ignore(..., ...)直到 '\n' . (您可以使用 <limits> 中的值来禁用字符计数检查并丢弃可能存在的任何字符数)

这留下了一个小问题。您正在循环中循环输入。如果用户输入 0要在此处表示要退出,则必须打破两个循环才能到达 "Goodbye!\n"; .虽然您可以设置标志等,但低级 goto仍然是跳出嵌套循环的标准且高效的方法。

总而言之,现在对您的问题有了更多了解,您可以执行类似以下操作:

#include <iostream>
#include <limits>

int main (void) {

    double n(0);

    do {
        int sum(0), n1(0);

        while (n1 == 0) {
            std::cout << "Enter a natural number or 0 for exit: ";
            std::cin >> n;
            if (!std::cin) {
                std::cin.clear();
                std::cin.ignore(std::numeric_limits<std::streamsize>::max(),
                                '\n');
            }
            else if (n == 0)
                goto alldone;
            else
                n1 = n;
        }

        for(int i = 1; i < n1; i++)
            if (n1 % i == 0)
                sum += i;

        if(n != int(n))
            std::cout << "Not a natural number!" << std::endl;
        else if(n1 < 0)
            std::cout << "Not a natural number!" << std::endl;
        else if(sum < n1)
            std::cout << "Number " << n << " is deficient!" << std::endl;
        else if(sum > n1)
            std::cout << "Number " << n << " is abudant!" << std::endl;
        else if(n1 && sum == n1)
            std::cout << "Number " << n << " is perfect!" << std::endl;

        sum = 0;

    } while(n != 0);

    alldone:;
    std::cout << "Goodbye!\n";

    return 0;
}

示例使用/输出

$ ./bin/inputnumber
Enter a natural number or 0 for exit: apples
Enter a natural number or 0 for exit: banannas
Enter a natural number or 0 for exit: 128
Number 128 is deficient!
Enter a natural number or 0 for exit: 96
Number 96 is abudant!
Enter a natural number or 0 for exit: foo
Enter a natural number or 0 for exit: 2
Number 2 is deficient!
Enter a natural number or 0 for exit: 0
Goodbye!

让我知道这是否是您遇到问题的地方。

关于c++ - 如何成功清除流?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49074048/

相关文章:

python - 如何使用 SWIG 为 C++ 模板类创建调度包装类

c++ - 如何将 std::ofstream& 传递给功能参数?

c++ - 用 std::forward_as_tuple 模拟 std::forward

c++ - 将限定的非静态成员函数作为函数指针传递

c++ - std::pow 的性能 - 缓存未命中?

C++:用变量推导。一小段代码有问题

c++ - 在 Visual Studio 调试命令参数中转义

c++ - 与 auto_ptr 声明不同,unique_ptr 声明是否在其模板类型为不完整类型时是明确定义的?

c++ - 是否可以在 cpp >= 11 中使用继承私有(private)构造函数

c++ - 是否有用于右值引用的 reference_wrapper<>?