【问题标题】:fflush(stdin) does not work compiled with gcc in cygwin but does compiled with visual studio 2010fflush(stdin) 不能在 cygwin 中使用 gcc 编译,但可以使用 Visual Studio 2010 编译
【发布时间】:2024-04-28 21:00:03
【问题描述】:

我正在(重新)学习编程,我从 C 开始。我的 IDE(如果我可以这么说的话)是 cygwin(32 位)和 Visual-Studio 2010,都在 Windows7 上。 我总是使用 gcc (cygwin) 以及 VS2010 编译器编译我编写的代码。我想,我这样做是因为我认为这是一种很好的学习方式。 无论如何,我刚刚了解了 fflush(stdin),即刷新标准输入缓冲区。似乎是一个很好的功能,因为否则,使用 scanf 似乎很痛苦。 所以我根据我教科书中的一个例子编写了下面的代码。它既可以用 cygwin 中的 gcc 编译,也可以用 VS2010 编译。当我运行 VS 编译程序时它工作正常(s.below),当我在 cygwin 中运行 gcc 编译程序时,fflush(stdin)不会刷新 stdin 缓冲区(s.below)。 我已经阅读了一些关于 fflush(stdin) 在某些情况下具有未定义行为的线程。不管这意味着什么,我都是从 C for Linux Programming 教科书中摘录的。如果 fflush(stdin) 不是清除标准输入缓冲区中内容的有效方法,还有什么其他标准方法?

非常感谢您的任何回答!

==程序在 Windows 下运行:

enter a long integer and a double
23.00 78.99
lint = 23
dt = 0.00

enter a five digits input
78493
u entered 78 and 493

==程序在 Cygwin 中运行:

enter a long integer and a double
23.00 78.99
lint = 23
dt = 0.00

enter a five digits input
u entered 78 and 2665720

==代码====

long lint;
double dt;
int fp, sp;
char fn[50], ln[50];

/* using the l modifier to enter long integers and doubles */
puts ("enter a long integer and a double");
scanf("%ld %lf", &lint, &dt);

printf("lint = %d\ndt = %.2f\n", lint, dt);

fflush(stdin); /*DOES NOT WORK UNDER CYGWIN!?*/

/* use field width to split input */

puts("\nenter a five digits input");
scanf("%2d %3d", &fp, &sp);

printf("u entered %d and %d\n", fp, sp);

【问题讨论】:

    标签: c visual-studio-2010 cygwin fflush


    【解决方案1】:

    7.21.5.2.2 的 C11 说(强调我的):

    如果stream 指向输出流或更新流,其中最近的 未输入操作,fflush 函数会导致该流的任何未写入数据 被交付到宿主环境中写入文件; 否则,行为未定义。

    这意味着您不应在输入流上调用fflush(除非您正在为定义了行为的非常特定的操作系统和库编写代码)

    这是有充分理由的!您通常会认为fflush(stdin) 会刷新您刚刚输入的行,对吗?好吧,还有更多。想象一下像这样运行你的程序:

    $ ./program < input_file
    

    在这种情况下,理论上所有文件都已经在stdin的缓冲区中。因此,刷新该缓冲区等于结束您的输入,这是一个非常无用的操作。由于这些原因,fflush 在输入流上不能有非常明智的行为,这就是为什么它在它们上未定义。


    如果你想忽略当前行,有更好的方法来做到这一点。一个例子如下:

    void flush_line(FILE *fin)
    {
        int c;
        do
        {
            c = fgetc(fin);
        } while (c != '\n' && c != EOF);
    }
    

    这会逐个字符地读取输入并停止,直到出现文件结束、读取错误或行结束。

    【讨论】:

      【解决方案2】:

      在标准中,没有为输入流定义 fflush()。在Microsoft's documentation 中明确定义。

      结果是,虽然您可以将它与具有明确定义行为的 Microsoft 的 C 库一起使用,但这种行为是不可移植的。

      如果您想使用 GCC 并进行这项工作,那么您可以使用 MinGW/GCC,因为它使用 Microsoft 的 C 运行时而不是 glibc。

      【讨论】:

      • +1 感谢它的工作原理。为什么 fflush(stdin) 有效但 flushall() 无效? pastebin.com/raw.php?i=ipcfQfRD
      • @barlop :我必须按照您的链接发现“不工作”是指“不链接”。在这个问题的上下文中,你的意思是它没有刷新输入缓冲区是一个合理的假设。也许您应该发布自己的单独问题?但是在 MS 库中,flushall() 已被弃用,取而代之的是 _flushall()
      • @barlop:好;但是下次发布一个问题 - 这不是 cmets 在 SO 上的目的。发布一个问题,您将获得更多的受众和更快的答案,以及添加对所有人有用的资源。你很幸运,在发布这个问题的答案两年后,我什至在附近阅读你的评论。
      【解决方案3】:

      You should not fflush an input stream。您观察到的差异来自此操作的行为未定义这一事实。

      【讨论】:

      • 您提供的链接和相关链接有一些很好的建议,但同时它们似乎是由没有 C 血统知识的人编写的。一些陈述是完全错误的或充其量是不准确的。小心点。
      【解决方案4】:

      据我了解,你正在为 Linux 编写这个程序,不管你信不信,fflush(stdin) 只能在 Windows 下工作。 (这至少是我从名为 C for dummies 的 C 书籍中得到的)。 Linux 上fflush(stdin) 的替代方案是fpurge(stdin)。试试看它是否适合你。

      【讨论】: