基于流的操作最终会调用read或者write函数进行I/O操作。为了使程序的运行效率最高,流对象通常会提供缓冲区,以减少调用系统I/O库函数的次数。
基于流的I/O提供以下3种缓冲:
全缓冲:直到缓冲区被填满,才调用系统I/O函数。对于读操作来说,直到读入的内容的字节数等于缓冲区大小或者文件已经到达结尾,才进行实际的I/O操作,将外存文件内容读入缓冲区;对于写操作来说,直到缓冲区被填满,才进行实际的I/O操作,缓冲区内容写到外存文件中。磁盘文件通常是全缓冲的。
行缓冲:直到遇到换行符'\n',才调用系统I/O库函数。对于读操作来说,遇到换行符'\n'才进行I/O操作,将所读内容读入缓冲区;对于写操作来说,遇到换行符'\n'才进行I/O操作,将缓冲区内容写到外存中。由于缓冲区的大小是有限的,所以当缓冲区被填满时,即使没有遇到换行符'\n',也同样会进行实际的I/O操作。标准输入stdin和标准输出stdout默认都是行缓冲的。
无缓冲:没有缓冲区,数据会立即读入或者输出到外存文件和设备上。标准出错stderr是无缓冲的,这样保证错误提示和输出能够及时反馈给用户,供用户排除错误。
以上3种缓冲区分别定义为3个宏,其定义如表21-1所示。
表21-1 缓冲区类型的宏定义
|
缓冲区类型 |
定 |
|
全缓冲 |
_IO_FULL_BUF |
|
行缓冲 |
_IO_LINE_BUF |
|
无缓冲 |
_IO_UNBUFFERED |
在使用上表所述的缓冲类型宏时,应将文件流对象中的缓冲区标志与该宏做"与"操作,判断结果是否为0即可知道该缓冲文件流的缓冲区是否属于该类型了。下面实例演示了得到文件流的缓冲区类型。该程序输出标准输出、标准输入和标准出错3个文件描述符的缓冲区类型、缓冲区大小等信息。
(1)在vi编辑器中编辑该程序如下:
程序清单21-1 buf.c 输出缓冲区的类型和缓冲区大小
#include <stdio.h> int main(void) return 0; |
$gcc buf.c -o buf |
$./buf stdout is line-buffered stderr is unbuffered |
$./buf < in.txt 1> out.txt 2> err.txt stdout is full-buffered stderr is unbuffered |
本文介绍了基于流的I/O操作中的三种缓冲机制:全缓冲、行缓冲和无缓冲,并通过示例程序展示了如何确定标准输入、标准输出及标准错误流的缓冲类型。

被折叠的 条评论
为什么被折叠?



