您的位置:首页 > 其它

标准I/O的三种缓冲

2016-01-26 16:33 330 查看
最近,有几个同学让我看了一个程序,后来发现问题都是出在使用printf()函数进行输出时的缓冲模式造成的。比如这样的程序:

#include
<stdio.h>

int main()

{
fprintf(stout,"1234");

while(1);

return 0;

}

程序并不是很长,但是就这样的程序出现了一个问题:为什么进入死循环之后,并没有输出“1234”,这个与我们预想的完全不一样,还以为是自己的程序写错了,可是看了又看,就三行代码,怎么会出错呢?原来这个就是标准I/O的缓冲模式造成的。
基于流的标准I/O的缓冲模式有三种:全缓冲,半缓冲,无缓冲。
全缓冲:直到缓冲区被填满,才调用系统I/O函数。对于读操作来说,直到读入的内容的字节数等于缓冲区大小或者文件已经到达结尾,才进行实际的I/O操作,将外存文件内容读入缓冲区;对于写操作来说,直到缓冲区被填满,才进行实际的I/O操作,缓冲区内容写到外存文件中。磁盘文件通常是全缓冲的。
行缓冲:直到遇到换行符'\n',才调用系统I/O库函数。对于读操作来说,遇到换行符'\n'才进行I/O操作,将所读内容读入缓冲区;对于写操作来说,遇到换行符'\n'才进行I/O操作,将缓冲区内容写到外存中。由于缓冲区的大小是有限的,所以当缓冲区被填满时,即使没有遇到换行符'\n',也同样会进行实际的I/O操作。标准输入stdin和标准输出stdout默认都是行缓冲的。
无缓冲:没有缓冲区,数据会立即读入或者输出到外存文件和设备上。标准出错stderr是无缓冲的,这样保证错误提示和输出能够及时反馈给用户,供用户排除错误。
看了这个应该明白了,原来这个是行缓冲造成的。我们是试着把
fprintf(stout,"1234");
改为
fprintf(stout,"1234\n");
这样的话,当我们陷入死循环之前,"1234"已经显示在标准输出上了。或者我们把

fprintf(stout,"1234");
改为
fprintf(stderr,"1234");

这个结果和上面的结果是一样的,只不过显示在了标准错误输出上了。
如果你不想按照上面的两种修改方式进行,也可以按照下面的方式,将I/O的缓冲设置为无缓冲方式。

#include
<stdio.h>

int main()

{

setbuf(stdout,
NULL);

fprintf(stout,
"1234");

while(1);

return 0;

}

这样修改之后的结果与前面两种方式修改之后的结果是一样的。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: