原创blog。转载请注明出处

头文件:

#include<unistd.h>

#include<sys/types.h>

函数原型:

pid_t fork( void);

(pid_t 是一个宏定义,事实上质是int 被定义在#include<sys/types.h>中)

返回值: 若成功调用一次则返回两个值。子进程返回0。父进程返回子进程ID。否则,出错返回-1

注意。子进程是父进程的副本,拷贝父进程的数据空间,堆栈等资源。

父子进程不共享上述资源。

每运行一次fork()函数,会返回两次。一次是在父进程,一次是在子进程,两次的返回值不一样。

我们来看一个简单的样例

#include<unistd.h>
#include<sys/types.h>
#include<stdio.h>
#include<stdlib.h>
int main(int argc, char ** argv )
{
pid_t result = fork();
if(result < 0)
{
printf("Error");
}
else if(result == 0)
{
printf("From the son");
}
else
{
printf("From the father");
}
}

输出

From the son
From the father

能够看到,父子进程都同一时候运行了这段代码。

能够这么理解,子进程拷贝父进程的全部代码。和堆栈,然后从fork()的下一行,子进程运行

再看一个样例

代码

#include<unistd.h>
#include<stdio.h>
#include<stdlib.h>
int main(int argc, char ** argv )
{
int i;
for(i = 0;i < 2;i++)
{
fork();
printf("%d\n",i);
}
}

然后我们编译运行

运行的结果是

[root@localhost test]# gcc -o first first.c
[root@localhost test]# ./first
0
1
1
0
1
1

为了便于分析,我们每次都输出当前进程的ppid(父进程)以及当前进程的pid

#include<unistd.h>
#include<stdio.h>
#include<stdlib.h>
int main(int argc, char ** argv )
{
int i;
for(i = 0;i < 2;i++)
{
fork();
printf("%d PPID:%d PID:%d\n",i,getppid(),getpid());
}
}

结果

[root@localhost test]# ./first
0 PPID:4984 PID:4985
1 PPID:4985 PID:4986
1 PPID:4984 PID:4985
0 PPID:4424 PID:4984
1 PPID:4984 PID:4987
1 PPID:4424 PID:4984

先分析PID。一共同拥有4种,所以一共同拥有4984,4985,4986,4987四个进程,所以4424是最開始的进程(main)的父进程,所以。最開始的进程是4984

然后根据pid来梳理进程的父子关系

依照子进程->父进程

4986->4985->4984->4424

4987->4984->4424

所以依照这个关系,我们绘图来分析比較直观

当中,输出的顺序依照红色圈中顺序来输出

由图能够看出,在我的这个CentOS系统中,子进程比父进程先运行,运行顺序在不同系统中不一样。所以编程的时候应当不依赖运行某一个系统的顺序。

然后,我们来分析一道经典的笔试面试题

     for(i = 0;i < 2;i++)
{
fork();
printf("%d\n",i);
}
}

  for(i = 0;i < 2;i++)
{
fork();
printf("%d",i);
}
}

各输出了几个0,几个1

对于前者

输出是

0
1
1
0
1
1

对于后者

输出是

01010101

对于前者,在上面已经进行了图解。所以不难理解。对于后者,不少同学会非常奇怪,为什么输出了4个0,4个1?

原因是:C语言中,printf函数假设遇到\n,会马上输出缓冲区内全部内容,假设没有\n,会先输出到缓冲区内。等待输出

我们继续绘图来分析,

watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvSGVsbG9fSHdj/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center" alt="">

这里在详解下:能够这么理解,每次printf的时候,是把本进程的printf缓冲区,增加到总的printf缓冲区(这样的说法可能不严谨)。比方到红色圈圈3的时候。4985本身在红色圈圈1的时候缓冲区已经有0了。那么再增加1,则本身进程的缓冲区为01,那么增加到printf总的缓冲区就是0101

原创blog,转载请注明出处

Linux fork函数具体图解-同一时候分析一道腾讯笔试题的更多相关文章

  1. linux fork函数与vfork函数,exit,_exit区别

    man vfork: NAME vfork - create a child process and block parent SYNOPSIS #include <sys/types.h> ...

  2. linux fork()函数

    C语言编程创建函数fork() 执行解析 | 浏览:1842 | 更新:2013-04-22 15:12 | 标签:c语言 概述 最近在看进程间的通信,看到了fork()函数,虽然以前用过,这次经过思 ...

  3. linux fork()函数 转载~~~~

    转自  ::  http://blog.csdn.net/jason314/article/details/5640969  一.fork入门知识 一个进程,包括代码.数据和分配给进程的资源.fork ...

  4. Linux—fork函数学习笔记

    fork()函数 在赋值语句pid = fork();之前,只有一个进程在执行这段代码,但在这条语句之后,就变成两个进程在执行了,这两个进程的代码部分完全相同.> 两个进程中,原先就存在的那个被 ...

  5. linux fork函数与vfork函数

    一.fork1. 调用方法#include <sys/types.h>#include <unistd.h> pid_t fork(void);正确返回:在父进程中返回子进程的 ...

  6. linux fork函数浅析

    #include <sys/types.h> #include <unistd.h> /* 功能:复制进程 參数:无 返回值: 成功: 父进程:返回子进程id 子进程:返回0 ...

  7. 《linux内核分析》第六周:分析fork函数对应的系统调用处理过程

    一. 阅读理解task_struct数据结构http://codelab.shiyanlou.com/xref/linux-3.18.6/include/linux/sched.h#1235: 进程是 ...

  8. Linux C 中 fork() 函数详解

    一.fork入门知识 一个进程,包括代码.数据和分配给进程的资源.fork() 函数通过系统调用创建一个与原来进程几乎完全相同的进程,也就是两个进程可以做完全相同的事,但如果初始参数或者传入的变量不同 ...

  9. linux的fork函数

       fork函数  头文件:#include<unistd.h> 函数原型:pid_t fork( void);(pid_t 是一个宏定义,其实质是int 被定义在#include< ...

随机推荐

  1. Leetcode 397.整数替换

    整数替换 给定一个正整数 n,你可以做如下操作: 1. 如果 n 是偶数,则用 n / 2替换 n.2. 如果 n 是奇数,则可以用 n + 1或n - 1替换 n.n 变为 1 所需的最小替换次数是 ...

  2. HDU-5319 Painter,深搜标记!

    Painter 题意:有一个棋盘n行,列数不超过50,用red和blue给这个棋盘涂色,每个格子每种颜色最多涂一次,如果两种颜色都涂了则该格子颜色为Green;red以斜杠'\'方式涂色,bule以' ...

  3. 九度oj 题目1172:哈夫曼树

    题目描述: 哈夫曼树,第一行输入一个数n,表示叶结点的个数.需要用这些叶结点生成哈夫曼树,根据哈夫曼树的概念,这些结点有权值,即weight,题目需要输出所有结点的值与权值的乘积之和. 输入: 输入有 ...

  4. 【Luogu】P1122最大子树和(DFS,树上DP)

    题目链接 感觉自己DP好烂啊   道道看题解 钦定1为根,DFS搜索子树权值.如果子树权值大于0就将当前节点加上子树权值,反之就把子树扔掉.最后在所有节点的权值中寻找最优解. void dfs(int ...

  5. 【Java工具】在代码头部加版权

    import java.io.File; import java.io.FileInputStream; import java.io.FileOutputStream; import java.io ...

  6. 实验三 kali下metasploit的漏洞攻击实践

    一.实验内容 1.使用kali进行靶机的漏洞扫描,利用metasploit选择其中的一个漏洞进行攻击,并获取权限. 2.分析攻击的原理以及获取了什么样的权限. 二.实验要求 1.熟悉kali原理和使用 ...

  7. Longge的问题(bzoj 2705)

    Description Longge的数学成绩非常好,并且他非常乐于挑战高难度的数学问题.现在问题来了:给定一个整数N,你需要求出∑gcd(i, N)(1<=i <=N). Input 一 ...

  8. 修改系统dpi

    系统dpi设置不合理,会导致屏幕显示效果差.配置系统dpi设置ro.sf.lcd_density的值即可,不同项目设置位置不同,以高通为例:需要修改 项目/device/qcom/common/roo ...

  9. springboot收集

    Spring Boot实战:拦截器与过滤器 参考:https://blog.csdn.net/m0_37106742/article/details/64438892 https://www.ibm. ...

  10. msp430项目编程52

    msp430综合项目---扩展项目二52 1.电路工作原理 2.代码(显示部分) 3.代码(功能实现) 4.项目总结