fork()的一道题

今天看博客的时候发现了这篇文章,作为最近有意向的的Linux,还是看了下,发现自己对fork()的理解确实是汗颜,硬是没弄明白,如下:


自己的分析稍后给出。。。。


下边原文:链接

#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个“_”。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值