https://blog.51cto.com/tonghuix/1165785
- stdout是行缓冲的,会先缓存到一个buffer,遇到\n才会输出。但是可以通过setvbuf接口修改其输出方式。
-
setvbuf(stdout, NULL, _IOFBF, 0)
- setvbuf说明
`setvbuf()` 是一个标准 C 库函数,用于设置一个已打开文件流(FILE *)的缓冲模式和缓冲区。这个函数允许你控制文件 I/O 的缓冲策略,以优化读写性能、减少系统调用次数或者满足特定的同步需求。其原型如下:
```c
int setvbuf(FILE *stream, char *buf, int mode, size_t size);
```
**参数说明:**
- `FILE *stream`: 指向已打开的文件流,通常通过 `fopen()`、`freopen()` 等函数获得。
- `char *buf`: 指向用户提供的缓冲区的指针。可以是 `NULL` 或指向一个已分配的缓冲区。
- 如果 `buf` 是 `NULL`,系统将自动分配和管理缓冲区。
- 如果 `buf` 不是 `NULL`,则使用提供的缓冲区。此时,`size` 参数必须指定缓冲区的大小。
- `int mode`: 指定缓冲模式,可以是以下常量之一:
- `_IOFBF` (Full buffering): 数据被积累到缓冲区满或者显式调用 `fflush()` 时才写入到磁盘。这是最常见的缓冲模式,适用于大多数常规文件操作,能减少磁盘 I/O 次数,提高效率。
- `_IOLBF` (Line buffering): 类似全缓冲,但在遇到换行符(`\n`)时,缓冲区内容也会被立即写入磁盘。常用于与终端交互的文本流,如标准输入(`stdin`)、标准输出(`stdout`)和标准错误(`stderr`)。
- `_IONBF` (No buffering): 不使用缓冲,每次对流的操作都直接对应一次系统调用。适合于需要立即写入或读取数据的场合,如日志文件、实时通信等。
- `size_t size`: 当 `buf` 不是 `NULL` 时,指定缓冲区的大小。如果 `buf` 是 `NULL`,并且 `mode` 为 `_IOFBF` 或 `_IOLBF`,`size` 被忽略,系统自动管理缓冲区大小。
**返回值:**
- 如果成功设置缓冲区,`setvbuf()` 返回 0。
- 如果失败(例如,缓冲区大小不合适、无法分配内存等),返回非零值。
**注意事项:**
1. `setvbuf()` 必须在第一次读写操作之前调用,即在 `fopen()` 之后、`fread()`、`fwrite()`、`fprintf()` 等之前调用。
2. 如果多次调用 `setvbuf()` 为同一文件流设置缓冲区,后续调用将覆盖前一次设置。
3. 对于标准输入、输出和错误流(`stdin`、`stdout`、`stderr`),其缓冲模式和缓冲区可能由系统默认设置,调用 `setvbuf()` 可能会改变这些默认设置。
总的来说,`setvbuf()` 函数允许程序员根据应用需求精细控制文件流的缓冲策略,以提高性能、适应特定同步需求或符合特定的编程约定。
#include <stdio.h>
#include <stdlib.h>
#define FILE_NAME "example.txt"
#define BUFFER_SIZE 4096
int main(void) {
FILE *file;
char *buffer;
/* 打开一个文件以进行读写 */
file = fopen(FILE_NAME, "w+");
if (file == NULL) {
fprintf(stderr, "Failed to open file '%s'\n", FILE_NAME);
return 1;
}
/* 分配用户提供的缓冲区 */
buffer = malloc(BUFFER_SIZE);
if (buffer == NULL) {
fprintf(stderr, "Failed to allocate buffer\n");
fclose(file);
return 1;
}
/* 设置缓冲模式为全缓冲,使用用户提供的缓冲区 */
if (setvbuf(file, buffer, _IOFBF, BUFFER_SIZE) != 0) {
fprintf(stderr, "Failed to set buffer using setvbuf()\n");
free(buffer);
fclose(file);
return 1;
}
/* 现在可以进行文件读写操作,它们将使用设置好的缓冲区 */
/* ... 这里放置你的文件读写代码 ... */
/* 完成文件操作后,记得释放缓冲区并关闭文件 */
free(buffer);
fclose(file);
return 0;
}