全缓冲和行缓冲

一、在linux系统中,标准的I/O提供了三种类型的缓冲。

1、全缓冲:在这种情况下,在填满I/O缓冲区后再进行实际的I/O操作。对于驻留在磁盘上的文件通常由标准I/O库实施全缓冲。调用fflush函数冲洗一个流。冲洗意味着将缓冲区的内容写到磁盘上。

2、行缓冲:在这种情况下,当在输入和输出遇到换行符时,标准I/O执行I/O操作。允许我们一次输出一个字符。涉及一个终端时,通常使用行缓冲。

对于行缓冲,有两个限制。第一,因为标准I/O库收集每一行的缓冲区的长度是固定的,所以只有填满的了缓冲区,那么即使没有换行符,也会进行I/O的操作。第二,任何时候只要通过标准I/O库要求从a一个布袋缓冲的流,或者b一个行缓冲的流(它要求从内核得到数据)得到输入数据,那么就会造成冲洗所有行缓冲输出流。在b中带了一个在括号中的说明,其理由是,所需的数据可能已在缓冲区中,他并不需求在需要数据时才从内核读数据。很明显,从不带缓冲的一个流中进行输入要求当时从内核得到数据。

3、不带缓冲的。标准I/O不对字符进行缓冲处理。例如:如果标准I/O函数fputs写15个字符到不带缓冲的流上,就会调用write的相关的函数立即写入打开的文件上。

二、通过下面的代码,可以更清楚的了解全缓冲和行缓冲的区别。

 

#include<stdio.h>
#include<unistd.h>
int glob=6;
char buf[]="a write ro stdout\n";
int main()
{
        int var;
		
        pid_t pid;
        printf("a write to stdout\n");
        //fflush(NULL);
        if((pid=fork())<0)
        {
                printf("fork error");
        }
        else
        {
                if(pid==0)
                {
                        glob++;
                        var++;
                }
                else
                {
                        sleep(2);
						//i++;
                }
        }
        printf("pid=%d,glob=%d,var=%d\n",getpid(),glob,var);
        exit(0);
}

编译后运行结果:

 

gcc buff.c -o buff

./buff


再者,运行./buff>temp

cat temp




第一种编译的时候,因为采用的是交互式终端,所以采用的是行缓冲 ,在printf之后立即刷新缓冲区。

第二种编译,将I/O重定向在temp文件中,查看temp的文件,会有两次输出a write to stdout,因为它采用的是全缓冲。

 

可以将程序中fflush(NULL)加入,则只会输出一次

 

a write to stdout

因为flush即时刷新了缓冲区


posted on 2013-10-11 15:15  love so much  阅读(759)  评论(0编辑  收藏  举报

导航