fork 和 vfork 的区别与联系
fork()函数:子进程是父进程的一个拷贝。即,子进程从父进程得到了数据段和堆栈段的拷贝,这些需要分配新的内存;而对于只读的代码段,通常使用共享内存的方式访问。fork返回后,子进程和父进程都从调用fork函数返回处开始执行。 父进程与子进程的不同之处在于:fork的返回值不同——父进程中的返回值为子进程的进程号,而子进程中fork的返回值为0 。
fork()函数的用途: 一个进程希望复制自身,从而父子进程能同时执行不同段的代码;进程想执行另外一个程序
vfork函数的用途: 用vfork创建的进程主要目的是用exec函数执行另外的程序;与fork的第二个用途相同
二者的区别:
1、数据共享:fork要拷贝父进程的数据段;而vfork则不需要完全拷贝父进程的数据段,在子进程没有调用exec和exit之前,子进程与父进程共享数据段。
2、父子进程运行顺序:fork不对父子进程的执行次序进行任何限制;而在vfork调用中,子进程先运行,父进程挂起,直到子进程调用了exec或exit之后,父子进程的执行次序才不再有限制。如果在调用这两个函数之前子进程依赖于父进程的进一步动作,则会导致死锁。
示例代码:
#include<sys/types.h>
#include<unistd.h>
#include<stdio.h>
#include<errno.h>
int main(int argc, char* grgv[])
{
pid_t pid;
int cnt=0;
printf("my process ID is %d\nnow to fork a process\n",getpid());
pid = vfork();
if(pid==0)
{
cnt++;
printf("cnt=%d\n",cnt);
// sleep(3);
printf("I am the child process, my process ID is %d\n",getpid());
_exit(0);
}else if(pid >0)
{
cnt++;
printf("cnt=%d\n",cnt);
printf("I am the parent process, my process ID is %d\n",getpid());
printf("my child process ID is %d\n",pid);
}else
{
printf("fork failed!\n");
printf("errorno is %d",errno);
}
}
注意:如果子进程中没有调用_exit(0), 则会报异常。
结束子进程不用exit(0),而使用_exit(0)。这是因为_exit(0)在结束进程时,不对标准I/O流进行任何操作。而exit(0)则会关闭进程的所有标准I/O流。