fork与vfork的区别(注:vfork子进程不能return)

1.vfork保证子进程先运行,在它调用exec或exit之后父进程才可能被调度运行。如果在调用这两个函数之前子进程依赖于父进程的进一步动作,则会导致死锁。

2.fork要拷贝父进程的进程环境;而vfork则不需要完全拷贝父进程的进程环境,在子进程没有调用exec和exit之前,子进程与父进程共享进程环境,相当于线程的概念,此时父进程阻塞等待。

 

为什么会有vfork呢?

因为以前的fork当它创建一个子进程时,将会创建一个新的地址空间,并且拷贝父进程的资源,然后将会有两种行为:

1.执行从父进程那里拷贝过来的代码段

2.调用一个exec执行一个新的代码段

 

当进程调用exec函数时,一个新程序替换了当前进程的正文,数据,堆和栈段。这样,前面的拷贝工作就是白费力气了,这种情况下,聪明的人就想出了vfork。vfork并不复制父进程的进程环境,子进程在父进程的地址空间中运行,所以子进程不能进行写操作,并且在儿子“霸占”着老子的房子时候,要委屈老子一下了,让他在外面歇着(阻塞),一旦儿子执行了exec或者exit后,相当于儿子买了自己的房子了,这时候就相当于分家了。

 

因此,如果创建子进程是为了调用exec执行一个新的程序的时候,就应该使用vfork

fork:拷贝了一份父进程的数据,也就是说父子之间互不干涉

vfork:与父进程共享同一份数据


程序参考:

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <unistd.h>
  4. int main(void) {
  5. int var;
  6. var = 88;
  7. pid_t pid;
  8. if ((pid = fork()) < 0) {
  9. printf( "vfork error");
  10. exit( -1);
  11. } else if (pid == 0) { /* 子进程 */
  12. var++;
  13. printf( "pid=%d,var=%d\n", getpid(), var);
  14. return 0;
  15. //exit(0);
  16. }
  17. printf( "pid=%d,var=%d\n", getpid(), var);
  18. return 0;
  19. }

运行结果:

  1. pid= 4684,var= 89
  2. pid= 4683,var= 88
因为此处用的是fork,所以父子之间互不干涉。

子进程中的var加一后,变为89。

而父进程中的var依然为88。


如果改为vfork,结果为:

  1. pid= 4785,var= 89
  2. pid= 4784,var= 89
因为共享数据了。


如果直接粘贴上述代码,只将fork改为vfork,会出现段错误。

原因是,在fork中用return语句是允许的。

因为子进程是复制了一份数据。

然而,在vfork中用return语句,因为父子共享,则会导致栈的崩溃。

也就是父进程不能够继续执行下去了。

因此,在vfork中需要用exit()函数。



猜你喜欢

转载自blog.csdn.net/iot_shun/article/details/80924598
今日推荐