浅谈Linux vfork与fork简单对比分析

2019-01-16 22:54:06王振洲

输出结果:

There is fork before

im father[3625],tmp is 5.

tmp = 6

im child[3626],tmp is 5.

tmp = 6

三、fork调用失败的原因:

①系统中已经存在太多进程,无法再创建新的进程。可通过ulimit -a命令查看当前所有的资源限制。

②内存不足,由于开辟每个新的进程都要分配一个PCB,并为新进程分配资源,内存都不足也就别提还想着再创建进程了。

vfork相关问题:

一、vfork基础了解

<1>vfork创建新进程的主要目的在于用exec函数执行另外的程序,实际上,在没调用exec或_exit之前子进程与父进程共享数据段。在vfork调用中,子进程先运行,父进程挂起,直到子进程调用exec或_exit,在这以后,父子进程的执行顺序不再有限制。

头文件:#include < unistd.h >

函数原型:pid_t vfork( )

返回值:返回值大于0则当前进程为父进程,等于0代表为子进程,小于零代表创建子进程失败。

通过一个例子来了解:

#include <stdio.h> #include <unistd.h> int tmp = 3; int main() { pid_t res = vfork(); if(res < 0){ perror("vfork"); _exit(); }else if(res == 0){ tmp = 10; printf("child res = %dn",tmp); _exit(0); }else{ printf("father res = %dn",tmp); } return 0; }

输出结果:

child res = 10

father res = 10

结果分析:正如上面所说的,子进程直接公用父进程的页表,改变子进程的数据也会影响到父进程。

这里写图片描述

<2>vfork用处:

vfork()跟fork()类似,都是创建一个子进程,这两个函数的的返回值也具有相同的含义。但是vfork()创建的子进程基本上只能做一件事,那就是立即调用_exit()函数或者exec函数族成员,调用任何其它函数(包括exit())、修改任何数据(除了保存vfork()返回值的那个变量)、执行任何其它语句(包括return)都是不应该的。更需要注意的是:调用vfork()之后,父进程会一直阻塞,直到子进程调用_exit()终止,或者调用exec函数族成员。

<3>为什么只能用_exit退出:

exit()是对_exit()的封装,它自己在调用_exit()前会做很多清理工作,其中包括刷新并关闭当前进程使用的流缓冲(比如stdio.h里面的printf等),由于vfork()的子进程完全共享了父进程地址空间,子进程里面的流也是共享的父进程的流,所以子进程里面是不能做这些事的。直接return就更不行了,子进程return以后,会从当前函数的外部调用点后面继续执行,这后面子进程可能将会执行很多语句,结果就没法预料了。在man手册中也强调了这一点,必须使用_exit退出。