c - 为什么将 stdout 重定向到 memfd_create() 结果仅在事先使用 stdout 时才有效?

标签 c linux

我在 64 位机器上运行 Ubuntu 20.04。

我想将 stdout 重定向到从 memfd_create 获得的描述符。似乎使用 memfd_create 创建的匿名文件只有在 stdoutdup2() 用于重定向之前获得非空输出时才会获取内容。为什么??

这是 SSCCU:

#define _GNU_SOURCE        
#include <sys/mman.h>
#include <sys/stat.h>
#include <stdio.h>
#include <unistd.h>

int main() {
    int temp_fd = memfd_create("temp", 0);
    struct stat sb;

    printf("a\n");
    dup2(temp_fd, 1);
    printf("b\n");
    fstat(temp_fd, &sb);
    fprintf(stderr, "%lu\n", sb.st_size);
}

这里,正如预期的那样,打印了 2,因为 b\n 到达匿名文件。

但是,如果我们删除 printf("a\n"); 然后打印 0。

为什么?

如果我们用 fflush(stdout)printf("") 替换 printf("a\n") 仍然打印 0 .

如果我们用 printf("a"); 替换,那么会按预期打印 3。

似乎必须在 dup2 之前将非空字符串打印到 stdout

编辑(回答后):memfd_create() 的相关性是什么?好吧,如果没有这个,输出文件就不是匿名的和易变的,所以我只是在之后手工检查了结果,一切都很好。

最佳答案

您的 libc 版本似乎会在第一次打印之前检查 FD 1 是否是终端。如果它是终端,它会设置行缓冲,因此每次打印换行符时都会刷新缓冲区。否则,完全缓冲,因此只有在满时才刷新。

这不是保证的行为。这正是您计算机上安装的 libc 版本所做的。

如果您想在任何特定时间刷新缓冲区,您可以使用 fflush(stdout)。或者您可以使用 setvbuf 来设置特定的缓冲模式。

关于c - 为什么将 stdout 重定向到 memfd_create() 结果仅在事先使用 stdout 时才有效?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/74061286/

相关文章:

xml - 如何通过 sed 或其他方式将文件中的 [^\u0009\u000A\u000D\u0020-\uD7FF\uE000-\uFFFD\u10000-\u10FFF]+ 替换为 ""?

c - 逐个获取 vsnprintf() 输出

java - 如何在C中读取字符串的第一个字符

linux - 为什么我的 Upstart 服务在重启后无法启动?

linux - Windows/Linux 中没有额外硬件的虚拟 COM

linux - 为什么 grep 返回不匹配的行

c - 协议(protocol) UDP 不支持的地址族 - C

c - scanf 字段宽度字符串溢出

c - 将静态库链接到可执行文件

linux - 将变量的内容写入文件