fork函数中的内存复制和共享
作者:网络转载 发布时间:[ 2014/12/29 17:24:18 ] 推荐标签:软件开发 .NET 函数
原来刚刚开始做Linux下面的多进程编程的时候,对于下面这段代码感到很奇怪:
#include<unistd.h>
#include<stdio.h>
#include<string.h>
#include<stdlib.h>
#include<stdarg.h>
#include<errno.h>
#define LEN 2
void err_exit(char*fmt,...);
int main(int argc,char*argv[])
{
pid_t pid;
int loop;
for(loop=0;loop<LEN;loop++)
{
if((pid=fork())<0)
err_exit("[fork:%d]:",loop);
else if(pid==0)
{
printf("Child process
");
}
else
{
sleep(5);
}
}
return 0;
}
为什么这段程序会创建3个子进程,而不是两个,为什么在第20行后面加上一个return 0;创建的又是两个子进程了?原来一直搞不明白,后来了解了C语言程序的存储空间布局以及在fork之后父子进程是共享正文段(代码段CS)之后才明白这其中的缘由!具体原理是啥,且容我慢慢道来!
首先得明白一个东西是C程序的存储空间布局,如下图所示:
(原图出自《UNIX环境高级编程》7.6节)
当一个C程序执行之后,它会被加载到内存之中,它在内存中的布局如上图,分为这么几个部分,环境变量和命令行参数、栈、堆、数据段(初始化和未初始化的)、正文段,下面挨个来说明这几段分别代表了什么:
环境变量和命令行参数:这些指的是Unix系统上的环境变量(比如$PATH)和传给main函数的参数(argv指针所指向的内容)。
数据段:这个是指在C程序中定义的全局变量,如果没有初始化,那么存放在未初始化的数据段中,程序运行时统一由exec赋值为0。否则存放在初始化的数据段中,程序运行时由exec统一从程序文件中读取。(了解汇编的朋友们想必知道汇编语言中的数据段DS,这和汇编中的数据段其实是一个东西)。
堆:这一部分主要用来动态分配空间。比如在C语言中用malloc申请的空间是在这个区域申请的。
正文段:C语言代码并不是直接执行的,而是被编译成了机器指令才能够在电脑上执行,终生成的机器指令是存放在这个区域(汇编中的代码段CS指的是这片区域)。
栈:个人感觉这是C程序内存布局关键的部分了。这个部分主要用来做函数调用。具体而言怎么说呢,程序刚开始栈中只有main这一个函数的内容(即main的栈帧),如果main函数要调用func函数,那么func函数的返回地址(main函数的地址),func函数的参数,func函数中定义的局部变量,还有func函数的返回值等等这些都会被压入栈中,这时栈中多了func函数的内容(func的栈帧)。然后func函数运行完了之后再来弹栈,把它原来压的内容去掉(即清除掉func栈帧),此时栈中又只剩下了main的栈帧。(这片区域是汇编中的栈段SS)
OK,这是C程序的存储器布局。这里我联想到另外一点,是全局变量和静态变量是存储在数据段中的,而局部变量是存储在栈中的,栈中数据在函数调用完之后一弹栈没了,这是为什么全局变量的生存周期比局部变量的生存周期要长的原因。
了解了C程序在存储器的布局之后,我们再来了解fork的内存复制机制,关于这个,我们只需要了解一句话够了,“子进程复制父进程的数据空间(数据段)、栈和堆,父、子进程共享正文段。”也是说,对于程序中的数据,子进程要复制一份,但是对于指令,子进程并不复制而是和父进程共享。具体来看下面这段代码(这是我在上面那段代码上稍微添加了一点东西):
/*这个程序会创建3个子进程,理解这句话,父子进程复制数据段、栈、堆,共享正文段
*
*/
#include<unistd.h>
#include<stdio.h>
#include<string.h>
#include<stdlib.h>
#include<stdarg.h>
#include<errno.h>
#define BUFSIZE 512
#define LEN 2
void err_exit(char*fmt,...);
int main(int argc,char*argv[])
{
pid_t pid;
int loop;
for(loop=0;loop<LEN;loop++)
{
printf("Now is No.%d loop:
",loop);
if((pid=fork())<0)
err_exit("[fork:%d]:",loop);
else if(pid==0)
{
printf("[Child process]P:%d C:%d
",getpid(),getppid());
}
else
{
sleep(5);
}
}
return 0;
}
为什么上面那段代码会创建三个子进程?我们来具体分析一下它的执行过程:
首先父进程执行循环,通过fork创建一个子进程,然后sleep5秒。
再来看父进程创建的这个子进程,这里我们记为子进程1.子进程1完全复制了这个父进程的数据部分,但是需要注意的是它的正文段是和父进程共享的。也是说,子进程1开始执行代码的部分并不是从main的{开始执行的,而是主函数执行到哪里了,它接着执行,具体而言是它会执行fork后面的代码。所以子进程1首先会打印出它的ID和它的父进程的ID。然后继续第二遍循环,然后这个子进程1再来创建一个子进程,我们记为子进程11,子进程1开始sleep。
子进程11接着子进程1执行的代码开始执行(即fork后面),它也是打印出它的ID和父进程ID(子进程1),然后此时loop的值再加1等于2了,所以子进程2直接返回了。
那个子进程1sleep完了之后也是loop的值加1之后变成了2,所以子进程1也返回了!
然后我们再返回去看父进程,它仅仅循环了一次,sleep完之后再来进行第二次循环,这次又创建了一个子进程我们记为子进程2。然后父进程开始sleep,sleep完了之后也结束了。
那么那个子进程2怎么样了呢?它从fork后开始执行,此时loop等于1,它打印完它的ID和父进程ID之后,结束循环了,整个子进程2直接结束了!
这是上面那段代码的运行流程,进程间的关系如下图所示:
上图中那个loop=%d是当这个进程开始执行的时候loop的值。上面那段代码的运行结果如下图:
这里这个3498进程是我们的主进程,3499是子进程1,3500是子进程11,3501是子进程2。
后,我们再来回答一下我们开始的时候提出的那个问题,为什么在子进程的处理部分“if(pid==0)”后加一个return 0,会创建两个子进程了,是因为子进程1运行到这里直接结束了,不再进行第二遍循环了,所以不会再去创建那个子进程11了,所以后一共是创建了两个子进程啊!
相关推荐
更新发布
功能测试和接口测试的区别
2023/3/23 14:23:39如何写好测试用例文档
2023/3/22 16:17:39常用的选择回归测试的方式有哪些?
2022/6/14 16:14:27测试流程中需要重点把关几个过程?
2021/10/18 15:37:44性能测试的七种方法
2021/9/17 15:19:29全链路压测优化思路
2021/9/14 15:42:25性能测试流程浅谈
2021/5/28 17:25:47常见的APP性能测试指标
2021/5/8 17:01:11