Linux fork函数具体图解-同一时候分析一道腾讯笔试题
原创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函数具体图解-同一时候分析一道腾讯笔试题的更多相关文章
- linux fork函数与vfork函数,exit,_exit区别
man vfork: NAME vfork - create a child process and block parent SYNOPSIS #include <sys/types.h> ...
- linux fork()函数
C语言编程创建函数fork() 执行解析 | 浏览:1842 | 更新:2013-04-22 15:12 | 标签:c语言 概述 最近在看进程间的通信,看到了fork()函数,虽然以前用过,这次经过思 ...
- linux fork()函数 转载~~~~
转自 :: http://blog.csdn.net/jason314/article/details/5640969 一.fork入门知识 一个进程,包括代码.数据和分配给进程的资源.fork ...
- Linux—fork函数学习笔记
fork()函数 在赋值语句pid = fork();之前,只有一个进程在执行这段代码,但在这条语句之后,就变成两个进程在执行了,这两个进程的代码部分完全相同.> 两个进程中,原先就存在的那个被 ...
- linux fork函数与vfork函数
一.fork1. 调用方法#include <sys/types.h>#include <unistd.h> pid_t fork(void);正确返回:在父进程中返回子进程的 ...
- linux fork函数浅析
#include <sys/types.h> #include <unistd.h> /* 功能:复制进程 參数:无 返回值: 成功: 父进程:返回子进程id 子进程:返回0 ...
- 《linux内核分析》第六周:分析fork函数对应的系统调用处理过程
一. 阅读理解task_struct数据结构http://codelab.shiyanlou.com/xref/linux-3.18.6/include/linux/sched.h#1235: 进程是 ...
- Linux C 中 fork() 函数详解
一.fork入门知识 一个进程,包括代码.数据和分配给进程的资源.fork() 函数通过系统调用创建一个与原来进程几乎完全相同的进程,也就是两个进程可以做完全相同的事,但如果初始参数或者传入的变量不同 ...
- linux的fork函数
fork函数 头文件:#include<unistd.h> 函数原型:pid_t fork( void);(pid_t 是一个宏定义,其实质是int 被定义在#include< ...
随机推荐
- Leetcode 375.猜数字大小II
猜数字大小II 我们正在玩一个猜数游戏,游戏规则如下: 我从 1 到 n 之间选择一个数字,你来猜我选了哪个数字. 每次你猜错了,我都会告诉你,我选的数字比你的大了或者小了. 然而,当你猜了数字 x ...
- zoj 2104 Let the Balloon Rise
Let the Balloon Rise Time Limit: 2 Seconds Memory Limit: 65536 KB Contest time again! How excit ...
- 公钥密码之RSA密码算法大素数判定:Miller-Rabin判定法!
公钥密码之RSA密码算法大素数判定:Miller-Rabin判定法! 先存档再说,以后实验报告还得打印上交. Miller-Rabin大素数判定对于学算法的人来讲不是什么难事,主要了解其原理. 先来灌 ...
- 九度oj 题目1019:简单计算器
题目描述: 读入一个只包含 +, -, *, / 的非负整数计算表达式,计算该表达式的值. 输入: 测试输入包含若干测试用例,每个测试用例占一行,每行不超过200个字符,整数和运算符之 ...
- tomcat的管理(manager)报错403
管理tomcat的时候遇到了以下问题: 1.刚开始需要用户名密码,不知道用户名和密码是什么,但是输入什么都不正确. 解决办法: 自己在tomcat-users.xml中按格式添加用户 conf文件夹里 ...
- 【Luogu】P3391文艺平衡树(Splay)
题目链接 ddosvoid和自为风月马前卒教了我这道题 他们好强啊 如果我们要反转区间[l,r] 我们首先把l的前驱旋转到根节点 再把r的后继旋转到根节点的右儿子 那么此时根节点的右儿子的左儿子所代表 ...
- 【HDOJ5978】To begin or not to begin(概率)
题意:有k个黑球和1个红球,两个轮流抽,抽到红球算赢,问先手赢的概率大还是后手大还是相等 k<=1e5 思路:手算前几项概率 大胆猜想 #include<cstdio> #inclu ...
- JavaScript 实现格式化字符串函数String.format (解决引号嵌套转义符问题)
在js开发中,我们可能会遇到这样一个问题 当需要通过js动态插入html标签的时候 特别是当遇到大量的变量拼接.引号层层嵌套的情况,会出现转义字符问题,经常出错 我们来看个例子 <!DOCTYP ...
- 安装配置JDK+Eclipse+Maven、Eclipse里新建Maven Project以及HDFS命令和Java API-课堂内容
步骤:1.安装JDK→2.安装Eclipse→3.安装Maven→4. Eclipse里配置Maven (下载Windows版本,在Windows里安装使用.) 1.安装配置JDK ①官网下载Java ...
- 批量修改WORD表格属性
有时候需要对word中很多表格的属性进行修改,而word无法批量修改属性,所有这里记录一个宏 Sub TableFormatter() Dim oTbl As Table, i As Integer ...