现在的位置: 首页 > 综合 > 正文

fork,printf缓冲

2017年12月05日 ⁄ 综合 ⁄ 共 1246字 ⁄ 字号 评论关闭

#include<stdio.h>

#include<sys/types.h>
#include<unistd.h>
int main(void)
{
    int i;
   for(i=0;i<2;i++)
    {
       fork();
       printf("_");
    }
    return 0;
}

#include<stdio.h>

#include<sys/types.h>
#include<unistd.h>
int main(void)
{
   int i;
   for(i=0;i<2;i++)
   {
      fork();
      printf("_\n");
   }
   return 0;
}
  fork()创建的新进程成为子进程。一次调用,两次返回,子进程的返回值是0,而父进程的返回值是新子进程的进程ID。
  C程序由正文段、初始化数据段、非初始化数据段、堆和栈组成。子进程获得父进程的数据段、堆和栈的副本,即复制而并非共享,并不包括正文段,但父子进程共享正文段
   !!!子进程和父进程继续执行fork()调用之后的指令
   标准I/O库提供缓冲的目的是尽可能的减少使用read和write调用的次数。
  标准I/O缓冲有三种:全缓冲、行缓冲和不带缓冲。行缓冲由换行符进行冲洗。
   腾讯笔试题之fork()

 

 第二层child执行时,复制父进程,此时缓冲区内无数据;因为第二层左子树父进程的执行时,因为没有换行符冲洗缓冲区,缓冲区内有一个“_”,此时第三层的fork()调用时,会复制此缓冲区,当执行完程序返回时,会冲洗缓冲区,即第三层第个子进程会输出两个“_”;同理,第三层第个子进程也会输出两个“_”。加起来有8个“_”。

   其实上图分支是假象,因为没有冲洗过缓冲区,故8个“_”分四次打印出来:第三层,第一个分支执行完打印2个“_”,因为其存入缓冲区2个“_”;第二个打印2个“_”,因为其自身产生一个“_”,同时冲洗复制父进程缓冲区而产生一个“_”第三个分支执行完打印2个“_”,因为第二层child压入缓冲区1个“_”,其左子树parent压入缓冲区一个“_”;第四个打印2个“_”,因为其自身产生一个“_”,同时冲洗复制父进程缓冲区而产生一个“_”。
    现在作假设:有第四层,将第三层分别编号为1、2、3、4,第四层第一个进程的左分支父进程为11,第一个进程的右分支子进程为12。11将压入缓冲区一个“_”,此时将打印三个“_”,12将复制之前的缓冲区,即复制两个“_”,加上自身产生一个“_”,也打印三个“_”。同理21也将压入缓冲区一个“_”,加上之前缓冲区的两个“_”,将打印三个;22将复制之前的缓冲区,即复制两个“_”,加上自身产生一个“_”,也打印三个“_”。加上31、32、41、42,一共8路,每路3个,将打印24个“_”。此节需认真领会。
 
腾讯笔试题之fork()
   因为有换行符冲洗缓冲区,故每个进程打印一次“_”。即6个“_”。但是,若不是交互式,将其执行结果输入至一个文件,就没有换行符冲洗缓冲区,此时将打印8个“_”。

【上篇】
【下篇】

抱歉!评论已关闭.