1

这段代码大部分都按预期工作,即提示用户输入单个字符,执行相关操作,提示用户按回车键,然后重复。但是,当我在提示符下输入 ^D (EOF) 时,会发生无限循环。我正在通过 std::cin.clear() 清除错误状态并调用 std::cin.ignore(...) 来清除缓冲区。什么可能导致无限循环?

#include <iostream>
#include <limits>

void wait()
{
    std::cout << std::endl << "press enter to continue.";
    std::cin.ignore(std::numeric_limits<std::streamsize>::max(), '\n');
    std::cin.clear();
    std::cin.get();
}

int main()
{
    char response;

    while (true)
    {
        std::cout << "enter a character at the prompt." << std::endl << "> ";
        std::cin >> response;
        switch (response)
        {
            case 'q':
                exit(0);
                break;
        }
        wait();
    }
}

如果重要的话,我正在 Mac OS X 终端中运行它。


更新:我在这里真正要问的是,当用户在提示符处输入 EOF (^D) 时,我如何 (a) 检测它并 (b) 重置流,以便用户可以继续输入数据。

以下示例与上面的代码不同,但说明了在检测到 ^D 后清除流并继续从该流中读取的相同原理。

> 一个
你输入了:
> 乙
你输入了:b
> ^D
你输入了EOF
> c
你输入了:c
...
4

7 回答 7

3

在调用格式化提取操作后,您应该始终检查是否设置了任何流的失败标志,在您的示例中,您正在检查response而不检查是否response正确提取。

此外,您std::endl在提示输出中使用它没有意义。std::endl打印\n然后刷新缓冲区,但是您会立即打印更多字符,因此刷新是多余的。由于cincout(通常)捆绑,调用输入函数在任何情况下std::cin都会导致std::cout被刷新,因此您最好将 a\n放入提示字符串并节省冗长的额外<<运算符。

为什么不创建一个提示函数来打印提示,检索输入并返回对流的引用,以便您可以使用通常的流到布尔类型转换来测试它是否成功。

这样你就可以摆脱 while true 和明确的中断。

std::istream& prompt_for_input( std::istream& in, std::ostream& out, char& response )
{
    out << "enter a character at the prompt.\n> ";
    in >> response;
    return in;
}

int main()
{
    char response;

    while ( prompt_for_input( std::cin, std::cout, response ) && response != 'q' )
    {
        wait();
    }
}
于 2009-10-21T06:09:24.857 回答
2

这个问题对标准输入没有真正意义。在该流结束后,将很难从标准输入中读取某些内容——您必须以某种方式重新打开它,但没有办法重新打开标准输入。它可能连接到管道、文件或终端——并且没有适合所有这些的行为。

因此,我假设您将从终端显式读取。在 UN*X 系统上,这意味着读取 /dev/tty,并在需要时重新打开它。这是一个简单的例子;大多数错误检查被省略。

// Warning: UN*X-specific

#include <iostream>
#include <fstream>

using namespace std;

int main()
{
    for(unsigned i=0; ; i++) {
        ifstream tty("/dev/tty");
        if (! tty) {
            cerr << "Failed to open TTY" << endl;
            return 2;
        }
        string s;
        while (getline(tty,s))
            cout << i << ": " << s << endl;
    }
    return 0;   // (unreached)
}
于 2009-10-21T09:46:59.130 回答
0
while ((std::cout << "Enter a character at the prompt ")
      && (!(std::cin >> response) || response =='q')) {
 std::cout << "Not a valid input";
 std::cin.clear();
 std::cin.ignore(std::numeric_limits<std::streamsize>::max(), '\n');

}

于 2009-10-21T09:28:50.677 回答
0

呃,我可能遗漏了一些东西,但我从来没有看到你break脱离while (true)循环。

// ...
while (true) {
    if (std::cin.eof()) {
        break;
    }
    // ...
}
于 2009-10-21T05:06:56.620 回答
0

如前所述,您需要确保流没有处于错误状态。我会更改 while 条件以使用 good()。不要只检查 EOF,因为除了 EOF 之外,流可以通过多种方式变得“坏”。

while (std::cin.good()) {...
于 2009-10-21T06:33:38.627 回答
0

读取 EOF 后,您只需忽略它并循环返回,而不退出循环,因此您将不断读取 EOF 并不断循环。如果你想在看到 EOF 时做一些事情,你需要在你的 switch 中或之前处理它。

也许您想在用户使用 ^D 关闭您的标准输入后从某处读取输入?在这种情况下,您必须关闭 cin 并重新打开它才能从您想要读取输入的其他地方读取。

于 2009-10-21T05:45:27.040 回答
0

您需要清除标志以使流在遇到 EOF 后执行大部分操作。

于 2009-10-21T03:11:09.693 回答