c - 在 gcc 中使用嵌套循环时 printf 不起作用

标签 c linux gcc printf

在使用 C 和 gcc 开发一个项目时,我发现了一些我无法理解的奇怪问题。下面的代码应该打印 i 的值,至少一次,但由于某种原因,在 Linux 和 gcc(还有 g++)上,它没有打印,它只是挂起而不输出任何内容。注意 printf 在其他场景下确实可以工作。

#include <stdio.h>

int main() {
    int i;
    int j;
    int z;
    for (i = 0; i < 47966; i++) {
        printf("%d ", i);
        for (j = 0; j < 47966; j++) {
            for (z = 0; z < 47966; z++) {
            }
        }
    }
    return 0;
}

有人遇到过这种情况吗?为什么会发生这种情况?

最佳答案

您可能是Suffering From Buffering

在大多数情况下,以大块写入设备比以大量小块写入设备更有效。写入文件句柄,以及 printf正在写信给stdout ,通常在发送到实际设备之前保存在内存缓冲区中。该缓冲区必须被“刷新”,这意味着缓冲区的内容被写入设备。缓冲分为三种类型:无缓冲、 block 缓冲和行缓冲。

stderr通常是无缓冲的,每次写入 stderr立即发送到设备。这很好,因为您希望立即看到错误信息。

文件通常是 block 缓冲的。写入内容存储在 BUFSIZ 的内存缓冲区中并在到达缓冲区或显式关闭文件句柄或进程终止时刷新。

stdout 行缓冲意味着它有一个缓冲区,但当它看到换行符或从 stdin 读取内容时会自动刷新。这是可用性和性能之间的折衷,通常您希望一次显示整行或在提示输入时显示整行。

您的程序永远不会输出换行符,因此您的数字将保留在行缓冲区中,直到缓冲区已满或进程终止。由于打印一个整数需要 2,300,737,156 次迭代,因此缓冲区需要一段时间才能被填满。在程序终止并自动刷新和关闭之前进行 110,357,158,424,696 次迭代 stdout 。不过,如果您使用优化来编译它,编译器会识别出内部循环不执行任何操作并消除它们;那么您的代码将非常快速地执行和打印。

您可以通过在使用 fflush(stdout) 打印后手动刷新缓冲区来解决此问题.

关于c - 在 gcc 中使用嵌套循环时 printf 不起作用,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/52487021/

相关文章:

c - 在 C 中将文件中的多个整数扫描到整数数组中

c - C 中的链接链表

c - 简单字符串操作中的段错误

Linux 2.6.24 内核编译错误 copy_user_generic_c 的 .size 表达式的计算结果不是常量

c++ - gpsd的交叉编译显示 “unrecognized option”

c++ - constexpr 使用静态函数初始化静态成员

c++ - 我需要帮助来理解这个 openMP 示例

Linux套接字和多线程程序

c++ - CUDA:错误:创建 thrust::device_ptr 时出现 "transfer of control bypasses initialization of"

gcc - cygwin64::./configure 返回 "configure: exit 77"