vfork

 vfork(建立一个新的进程)
  相关函数 wait,execve
  头文件 #include<unistd.h>
  定义函数 pid_t fork(void);

函数说明

  vfork()会产生一个新的子进程.但是vfork创建的子进程与 父进程 共享 数据段 ,而且由vfork创建的
  子进程将先于 父进程 运行.fork()的使用详见百度词条fork().
  vfork()用法与fork()相似.但是也有区别,具体区别归结为以下3点:
  1. fork():子进程拷贝父进程的 数据 段, 代码段 . vfork():子进程与父进程共享 数据 段.
  2. fork():父子进程的执行次序不确定.
  vfork():保证子进程先运行,在调用exec或exit之前与 父进程 数据是共享的,在它调用exec
  或exit之后 父进程 才可能被调度运行。
  3. vfork()保证子进程先运行,在她调用exec或exit之后 父进程 才可能被调度运行。如果在
  调用这两个 函数 之前子进程依赖于 父进程 的进一步动作,则会导致死锁。
  下面通过几个例子加以说明:


返回值

  如果vfork()成功则在 父进程 会返回新建立的子进程代码(PID),
  而在新建立的子进程中则返回0。如果vfork失败则直接返回-1,失
  败原因存于errno中。
   错误代码
  EAGAIN 进程数已达系统规定上限
  ENOMEM  内存不足 ,无法配置核心所需的 数据结构 空间。


范例

   #include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
int main(void)
{
 pid_t pid;
 if((pid = vfork()) < 0)
 {
  perror("vfork is failed");
  exit(1);
 }else if(pid == 0)
 {
  printf("this is child process\n");
  exit(0);
 }else
 {
  printf("this is parent process\n");
  if(wait(NULL) == -1)
   {
     perror("fail to wait");
     exit(1);
   }
 }
 return 0;
}
   执行
   this is child process
   this is parent process
  注:如果在程序中不加exit语句,程序会一直循环执行下去,直到进程号被分配完才会退出


vfork与fork的区别

  vfork用于创建一个新进程,而该新进程的目的是exec一个新进程,vfork和fork一样都创建一个子进程,但是它并不将父进程的地址空间完全复制到子进程中,不会复制页表。因为子进程会立即调用exec,于是也就不会存放该地址空间。不过在子进程中调用exec或exit之前,他在父进程的空间中运行。
  为什么会有vfork,因为以前的fork当它创建一个子进程时,将会创建一个新的地址空间,并且拷贝父进程的资源,而往往在子进程中会执行exec调用,这样,前面的拷贝工作就是白费力气了,这种情况下,聪明的人就想出了vfork,它产生的子进程刚开始暂时与父进程共享地址空间(其实就是线程的概念了),因为这时候子进程在父进程的地址空间中运行,所以子进程不能进行写操作,并且在儿子“霸占”着老子的房子时候,要委屈老子一下了,让他在外面歇着(阻塞),一旦儿子执行了exec或者exit后,相当于儿子买了自己的房子了,这时候就相当于分家了。
  vfork和fork之间的另一个区别是: vfork保证子进程先运行,在她调用exec或exit之后父进程才可能被调度运行。如果在调用这两个函数之前子进程依赖于父进程的进一步动作,则会导致死锁。由此可见,这个系统调用是用来启动一个新的应用程序。其次,子进程在vfork()返回后直接运行在父进程的栈空间,并使用父进程的内存和数据。这意味着子进程可能破坏父进程的数据结构或栈,造成失败。
  为了避免这些问题,需要确保一旦调用vfork(),子进程就不从当前的栈框架中返回,并且如果子进程改变了父进程的数据结构就不能调用exit函数。子进程还必须避免改变全局数据结构或全局变量中的任何信息,因为这些改变都有可能使父进程不能继续。通常,如果应用程序不是在fork()之后立即调用exec(),就有必要在fork()被替换成vfork()之前做仔细的检查。
  用fork函数创建子进程后,子进程往往要调用一种exec函数以执行另一个程序,当进程调用一种exec函数时,该进程完全由新程序代换,而新程序则从其main函数开始执行,因为调用exec并不创建新进程,所以前后的进程id 并未改变,exec只是用另一个新程序替换了当前进程的正文,数据,堆和栈段。


附加:
相信大家对fork()都很熟了,不过有一点关于fork()的我想说名一下:
    现在的Linux内核实现fork函数时往往实现为子进程先于父进程共享代码,数据段和堆栈,当子进程修改这些共享内容时,复制才会发生,内核才会给子进程分配进程空间将父进程内容复制过来,继续后面的操作。这样实现更加合理,对于一些只是为复制自身完成一些工作的进程来说,这样做效率会更高。这是现代操作系统中一个重要的概念----“写时复制”的一个重要体现。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值