通过fork函数解读linux系统行缓冲与全缓冲

       基于流的操作最终会调用read或者write函数进行I/O操作。为了使程序的运行效率最高,流对象通常会提供缓冲区,以减少调用系统I/O库函数的次数。

1、全缓冲 。全缓冲指的是系统在填满标准IO缓冲区之后才进行实际的IO操作;注意,对于驻留在磁盘上的文件来说通常是由标准IO库实施全缓冲。

2、行缓冲 。在这种情况下,标准IO在输入和输出中遇到换行符时执行IO操作;注意,当流涉及终端的时候,通常使用的是行缓冲。

3、无缓冲 。无缓冲指的是标准IO库不对字符进行缓冲存储;注意,标准出错流stderr通常是无缓冲的。

fork函数:类unix系统中,一个现有的进程可以调用fork函数创建一个新的进程。由fork创建的新进程成为子进程(child process)。fork函数被调用一次,但是返回两次。两次返回的唯一区别是子进程的返回值是0,而父进程的返回值则是新子进程的进程ID。子进程和父进程分别继续执行fork调用之后的命令。子进程获得父进程数据空间、堆、栈(以及IO缓冲区)的副本,父子进程共享正文段。

          下面是圣经(UNIX高级环境编程)中fork函数的示例:

  1#include <unistd.h>
  2 #include <stdlib.h>
  3 #include <stdio.h>
  4 int glob=6;
  5 char buf[]="a write to stdout\n";
  6 int main(void){
  7    int var;
  8    pid_t pid;
  9    var=88;
 10    if(write(1,buf,sizeof(buf)-1)!=sizeof(buf)-1){
 11       perror("write");
 12       return -1;
 13    }
 14    printf("before fork\n");
 15    if((pid=fork())<0){
 16       perror("fork\n");
 17       return -1;
 18    }else if(pid==0){
 19       glob++;
 20       var++;
 21    }else
 22     sleep(1);
 23    printf("pid=%d,glob=%d,var=%d\n",getpid(),glob,var);
 24    exit(0);
 25 }

执行程序:

UC#./a.out
a write to stdout
before fork
pid=6422,glob=7,var=89
pid=6421,glob=6,var=88

重定向再查看重定向文件:

UC#./a.out >q
UC#cat q
a write to stdout
before fork
pid=6424,glob=7,var=89
before fork
pid=6423,glob=6,var=88

前后对比发现,重定向到文件再输出时,多了一行通过printf打印的信息,这是为什么呢?

原因便在于前面提到的行缓冲与全缓冲,当write到1(标准输出)时,用的是行缓冲,遇到‘\n'便输出缓冲区内容,而写文件时则是全缓冲,要到缓冲区满或进程结束时才写入到文件。

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值