先看一个fork的例子:

int glob = ;

int main(void)
{
int var, pid;
var = ; if ((pid = fork()) < ) {
printf("vfork error");
exit(-); } else if (pid == ) { /* 子进程 */
var++;
glob++;
exit();
} printf("pid=%d, glob=%d, var=%d\n", getpid(), glob, var); return ;
}

运行结果:

[root@localhost tmp]# ./a.out
pid=, glob=, var=

可见,子进程修改的局部变量var和全局变量glob后,父进程是不可见的。

如果把代码中的fork替换成vfork,再次运行,得到的结果:

[root@localhost tmp]# ./a.out
pid=, glob=, var=

可见,父进程共享了子进程的修改操作。

在使用vfork时,如果子进程使用return语句结束,会发生什么呢?

int glob = ;

int main(void)
{
int var, pid;
var = ; if ((pid = vfork()) < ) {
printf("vfork error");
exit(-); } else if (pid == ) { /* 子进程 */
var++;
glob++;
return ;
} printf("pid=%d, glob=%d, var=%d\n", getpid(), glob, var); return ;
}

在我的机器上,导致了无限循环(直到vfork调用出错),这是因为子进程调用return语句破坏了父进程的栈。

fork与vfork的区别:

  • fork 是 创建一个子进程,并把父进程的地址空间copy到子进程中;
  • vfork是 创建一个子进程,并和父进程的地址空间share一起用。

我们知道,fork通常采用写时复制技术(copy-on-write, COW)创建子进程,以提高进程clone的性能;但在更早还没有COW的年代,fork创建子进程时时需要完整的复制父进程地址空间到子进程中,如果我们创建子进程的目的是为了调用exec,那么这种复制就显得既低效又无必要。而vfork让子进程共享父进程的地址空间,而不作克隆操作,就是为了节省这种不必要的复制开销。

回到上面return导致程序crash的例子,return会释放局部变量,并弹栈,回到上级函数执行。exit直接退掉。如果你用c++ 你就知道,return会调用局部对象的析构函数,exit不会。(注:exit不是系统调用,是glibc对系统调用 _exit()或_exitgroup()的封装)

可见,子进程调用exit() 没有修改函数栈,所以,父进程得以顺利执行。而子进程调用return,相当于在父进程的栈上执行了弹栈操作,父进程也就跪了。

注意:

1、vfork保证子进程先运行,在它调用exec或exit之后父进程才可能被调度运行;

2、子进程在调用exec或exit之前是在父进程的地址空间中运行的。

可见,vfork的设计初衷是为了应对那些子进程需要马上调用exec的场景,因此不对父进程的地址空间做任何复制。


再看一个fork的有趣例子,

int main(void)
{
int i, pid = ; for (i = ; i < ; i++) { pid = fork(); if (pid == ) {
printf("pid:%d\n", getpid());
}
} return ;
}

问题是,执行这段代码,一共产生了几个进程呢?

从执行结果来看,printf函数打印了3次,fork被调用了3次,连上main进程一共有4个进程。

再看下面这个例子,一共打印了多少个 “_” 呢?

int main(void)
{
int i; for(i=; i<; i++){ fork(); printf("-"); } return ;
}

按照上面的例子,程序运行过程中一共有4个进程,把main进程记为A,则有

i=0时,A进程 fork调用,产生子进程B1,然后A、B1各打印一个"_";

i=1时,A进程 fork调用,产生子进程B2,然后A、B2各打印一个"_";

与此同时,B1进程fork调用,产生子进程C1,然后B1、C1各打印一个"_";

看起来,好像有6个"_"被打印了,但这段代码的执行结果却是8个,这是为啥呢?

先来看下,这4个进程间的关系如下:

A --> B1  --> C1

|--> B2

可见,B1、B2继承自A,而C1继承自B1。

1、B1是在i=0时复制A的,此时A还没有调用过printf函数;

2、B2是在i=1时复制A的,此时A已经调用过一次printf函数;

3、C1是在i=1时复制B1的,此时B1已经调用过一次printf函数;

我们知道,fork进程会让子进程完整复制父进程的地址空间,这也就包括了I/O缓冲区,这就是为什么最终打印了8个"_"的原因。

参考文档:

http://coolshell.cn/articles/12103.html

http://coolshell.cn/articles/7965.html

fork与vfork的更多相关文章

  1. fork、vfork、clone区别

    在Linux中主要提供了fork.vfork.clone三个进程创建方法. 问题 在linux源码中这三个调用的执行过程是执行fork(),vfork(),clone()时,通过一个系统调用表映射到s ...

  2. Linux下fork()、vfork()、clone()和exec()的区别

    转自Linux下fork().vfork().clone()和exec()的区别 前三个和最后一个是两个类型.前三个主要是Linux用来创建新的进程(线程)而设计的,exec()系列函数则是用来用指定 ...

  3. c语言exit和return区别,在fork和vfork中使用

    转自c语言exit和return区别,在fork和vfork中使用 exit函数在头文件stdlib.h中. 简述: exit(0):正常运行程序并退出程序: exit(1):非正常运行导致退出程序: ...

  4. 进程创建函数fork()、vfork() ,以及excel()函数

    一.进程的创建步骤以及创建函数的介绍 1.使用fork()或者vfork()函数创建新的进程 2.条用exec函数族修改创建的进程.使用fork()创建出来的进程是当前进程的完全复制,然而我们创建进程 ...

  5. fork与vfork详解

    一.fork函数 要创建一个进程,最基本的系统调用是fork,系统调用fork用于派生一个进程,函数原型如下: pid_t fork(void)  若成功,父进程中返回子进程ID,子进程中返回0,若出 ...

  6. linux 进程创建clone、fork与vfork

    目录: 1.clone.fork与vfork介绍 2.fork说明 3.vfork说明 4.clone说明5.fork,vfork,clone的区别 内容: 1.clone.fork与vfork介绍 ...

  7. 1.2 Linux中的进程 --- fork、vfork、exec函数族、进程退出方式、守护进程等分析

    fork和vfork分析: 在fork还没有实现copy on write之前,Unix设计者很关心fork之后立即执行exec所造成的地址空间浪费,也就是拷贝进程地址空间时的效率问题,所以引入vfo ...

  8. UNIX高级环境编程(9)进程控制(Process Control)- fork,vfork,僵尸进程,wait和waitpid

    本章包含内容有: 创建新进程 程序执行(program execution) 进程终止(process termination) 进程的各种ID   1 进程标识符(Process Identifie ...

  9. fork()、vfork()、clone()和exec()

    前三个和最后一个是两个类型.前三个主要是Linux用来创建新的进程(线程)而设计的,exec()系列函数则是用来用指定的程序替换当前进程的所有内容.所以exec()系列函数经常在前三个函数使用之后调用 ...

随机推荐

  1. codeforces Hill Number 数位dp

    http://www.codeforces.com/gym/100827/attachments Hill Number Time Limits:  5000 MS   Memory Limits: ...

  2. Numpy 用法小结

    1.  asarray 函数 可以将输入数据转化为矩阵格式. 输入数据可以是(列表,元组,列表的列表,元组的元组,元组的列表等这些数组形式). >>> asarray([(1,2,3 ...

  3. bzoj 2938 AC自动机

    根据题意建出trie图,代表单词的点不能走,直接或间接指向它的点也不能走.这样的话如果能在图中找到一个环的话就是TAK,否则是NIE. #include<iostream> #includ ...

  4. php热身2:CRUD with Ajax

    这次热身是一个会员管理系统,包括会员注册.登录.资料修改功能,使用ajax技术 1.建表 use common_module; create table if not exists member( u ...

  5. 自动完成--autoComplete插件(2)

    远端的也可以成为本地的数据 4) lookupLimit 类型:数字 说明:本地数据显示的最大条数,服务器段的没有效果,服务器端的可以后台设置 默认:没有限制 5) lookupFilter 类型: ...

  6. 【poj1061】 青蛙的约会

    http://poj.org/problem?id=1061 (题目链接) 题意 两只青蛙在周长为L的球上沿一条直线向一个方向跳,每只每次分别跳m,n米,它们一开始分别在X,Y处,问跳几次两青蛙可以在 ...

  7. Cloud Design Patterns Book Reading(undone)

    目录 . the most common problem areas in cloud application development ) Availability ) Data Management ...

  8. easyui使用datagrid时列名包含特殊字符导致表头与数据错位的问题

    做一个用easyui的datagrid显示数据的功能时发现表格的列头与数据错位了,而且这个现象不总是能重现,一直没搞清楚原因.后来偶然在控制台看出了一点端倪: 推测表头或者单元格的class名应该是用 ...

  9. HDU 5726 GCD

    传送门 GCD Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others) Problem ...

  10. codevs 1432 总数统计

    1432 总数统计 时间限制: 1 s空间限制: 128000 KB题目等级 : 钻石 Diamond   题目描述 Description 给出n个数,统计两两之和小于k的方案数之和. 输入描述 I ...