AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • 主页
  • 系统&网络
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • 主页
  • 系统&网络
    • 最新
    • 热门
    • 标签
  • Ubuntu
    • 最新
    • 热门
    • 标签
  • Unix
    • 最新
    • 标签
  • DBA
    • 最新
    • 标签
  • Computer
    • 最新
    • 标签
  • Coding
    • 最新
    • 标签
主页 / unix / 问题 / 786352
Accepted
SekstusEmpiryk
SekstusEmpiryk
Asked: 2024-11-09 18:36:03 +0800 CST2024-11-09 18:36:03 +0800 CST 2024-11-09 18:36:03 +0800 CST

读取输入时自动刷新流

  • 772

我在《Unix环境高级编程》中读到过这样的内容:

行缓冲有两个注意事项。首先,标准 I/O 库用于收集每行的缓冲区大小是固定的,因此如果我们在写入换行符之前填充此缓冲区,则可能会发生 I/O。其次,每当通过标准 I/O 库从 (a) 无缓冲流或 (b) 行缓冲流(需要从内核请求数据)请求输入时,所有行缓冲输出流都会被刷新。 (b) 上限定符的原因是请求的数据可能已经在缓冲区中,这不需要从内核读取数据。显然,任何来自无缓冲流的输入(项目 (a))都需要从内核获取数据。

https://www.gnu.org/software/libc/manual/html_node/Flushing-Buffers.html指出:

刷新缓冲流上的输出意味着将所有累积的字符传输到文件。在许多情况下,流上的缓冲输出会自动刷新:

当您尝试进行输出并且输出缓冲区已满时。

当流关闭时。请参阅关闭流。

当程序通过调用 exit 终止时。请参阅正常终止。当写入换行符时(如果流是行缓冲的)。

每当任何流上的输入操作实际上从其文件读取数据时。

所以我编写了一个程序来测试这个:

#include <stdio.h>
#include <unistd.h>

int
main()
{
    printf("Hello");
    FILE *fp = fopen("hugefile", "r");
    if (fp == NULL) {
        fprintf(stderr, "Cannot open file\n");
        return -1;
    }
    setvbuf(fp, NULL, _IONBF, 0);
    sleep(2);
    char buf[10];

    while (fread(buf, sizeof(char), 10, fp) == 10) {
    }

    if (ferror(fp)) {
        fprintf(stderr, "Read error\n");
        return -1;
    }; // I expect "Hello" to appear on a screen but it doesn't
    fclose(fp);
    sleep(50);
}

但 fread() 不会触发输出闪烁。我是不是漏掉了什么?

我不知道这里可能与什么相关但我使用:

gcc 版本 14.2.1 20240910

glibc 2.40+r16+gaa533d58ff-2

c
  • 1 1 个回答
  • 74 Views

1 个回答

  • Voted
  1. Best Answer
    Arkadiusz Drabczyk
    2024-11-09T19:47:57+08:002024-11-09T19:47:57+08:00

    我认为刷新缓冲区并不意味着它会被打印到屏幕上或者任何标准输出被重定向到的地方。

    通过读取标准输入来刷新标准输出:

    您可以在 buf.c 中实现以下行为:

    #include <stdio.h>
    #include <unistd.h>
    
    int main(void)
    {
      printf("Hello");
      sleep(2);
      fprintf(stderr, "No output from stdout after 2 seconds\n");
    
      char buf[10];
      while (fread(buf, sizeof(char), 10, stdin) == 10) {
      }
    
      sleep(5);
    }
    

    运行它:

    $ ./buf
    No output from stdout after 2 seconds
    Hello
    

    “Hello”字符串打印到 stdout 并且不以换行符结尾)(\n),因此它不会在 2 秒后刷新,但它会在 fread()运行时刷新。如果删除它,它将仅在最后一次睡眠后刷新。

    每当任何流上的输入操作实际上从其文件读取数据时。

    通过读取文件刷新到文件:

    重现每当任何流上的输入操作实际上从其文件读取数据时。我认为这是你的主要目标:

    #include <stdio.h>
    #include <unistd.h>
    
    int main(void)
    {
      FILE *fp = fopen("FILE", "a+");
      fprintf(fp, "add more characters to FILE\n");
      fprintf(stderr, "wrote more characters to FILE\n");
      sleep(5);
    
      fprintf(stderr, "reading will now start\n");
      fseek(fp, 0, SEEK_SET);
      char buf[10];
      while (fread(buf, sizeof(char), 10, fp) == 10) {
    
      }
    
      fprintf(stderr, "Sleep for 5 seconds before sleep()\n");
      sleep(5);
      fprintf(stderr, "Exit\n");
    }
    

    像这样运行它,我将得到一个仅包含当前日期的小文件,并在后台运行一个二进制文件,以便我能够看到它打印的所有消息并能够同时检查文件的内容:

    $ date > FILE
    $ ./buf &
    [1] 14522
    wrote more characters to FILE
    $ cat FILE
    Sat Nov  9 13:02:28 CET 2024
    $ reading will now start
    Sleep for 5 seconds before sleep()
    
    $ cat FILE
    Sat Nov  9 13:02:28 CET 2024
    add more characters to FILE
    Exit
    

    如您所见,只有在运行add more characters to FILE后才将行添加到FILE中。fread()

    • 2

相关问题

  • 从 objdump 获取仅十六进制的输出

  • 使用 read() 读取 /proc/pid/maps

  • 堆什么时候用于动态内存分配?

  • GNU make dep 究竟做了什么?

  • 基于 OpenCV 的程序优化嵌入式 linux OS [关闭]

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    模块 i915 可能缺少固件 /lib/firmware/i915/*

    • 3 个回答
  • Marko Smith

    无法获取 jessie backports 存储库

    • 4 个回答
  • Marko Smith

    如何将 GPG 私钥和公钥导出到文件

    • 4 个回答
  • Marko Smith

    我们如何运行存储在变量中的命令?

    • 5 个回答
  • Marko Smith

    如何配置 systemd-resolved 和 systemd-networkd 以使用本地 DNS 服务器来解析本地域和远程 DNS 服务器来解析远程域?

    • 3 个回答
  • Marko Smith

    dist-upgrade 后 Kali Linux 中的 apt-get update 错误 [重复]

    • 2 个回答
  • Marko Smith

    如何从 systemctl 服务日志中查看最新的 x 行

    • 5 个回答
  • Marko Smith

    Nano - 跳转到文件末尾

    • 8 个回答
  • Marko Smith

    grub 错误:你需要先加载内核

    • 4 个回答
  • Marko Smith

    如何下载软件包而不是使用 apt-get 命令安装它?

    • 7 个回答
  • Martin Hope
    user12345 无法获取 jessie backports 存储库 2019-03-27 04:39:28 +0800 CST
  • Martin Hope
    Carl 为什么大多数 systemd 示例都包含 WantedBy=multi-user.target? 2019-03-15 11:49:25 +0800 CST
  • Martin Hope
    rocky 如何将 GPG 私钥和公钥导出到文件 2018-11-16 05:36:15 +0800 CST
  • Martin Hope
    Evan Carroll systemctl 状态显示:“状态:降级” 2018-06-03 18:48:17 +0800 CST
  • Martin Hope
    Tim 我们如何运行存储在变量中的命令? 2018-05-21 04:46:29 +0800 CST
  • Martin Hope
    Ankur S 为什么 /dev/null 是一个文件?为什么它的功能不作为一个简单的程序来实现? 2018-04-17 07:28:04 +0800 CST
  • Martin Hope
    user3191334 如何从 systemctl 服务日志中查看最新的 x 行 2018-02-07 00:14:16 +0800 CST
  • Martin Hope
    Marko Pacak Nano - 跳转到文件末尾 2018-02-01 01:53:03 +0800 CST
  • Martin Hope
    Kidburla 为什么真假这么大? 2018-01-26 12:14:47 +0800 CST
  • Martin Hope
    Christos Baziotis 在一个巨大的(70GB)、一行、文本文件中替换字符串 2017-12-30 06:58:33 +0800 CST

热门标签

linux bash debian shell-script text-processing ubuntu centos shell awk ssh

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve