设为首页 收藏本站
查看: 868|回复: 0

[经验分享] Linux之fork与vfork区别

[复制链接]
累计签到:1 天
连续签到:1 天
发表于 2016-7-21 08:47:30 | 显示全部楼层 |阅读模式
创建一个新进程的方法只有由某个已存在的进程调用fork()或vfork()
wKiom1ePL3ez9NYFAAAHVJ_16P4891.jpg
1.fork()函数
wKioL1ePL5vxnPw7AAAepqgYMns615.jpg
返回值:成功:父进程:返回子进程的PID
                       子进程:返回0
             失败:父进程返回-1
子进程是父进程的一个拷贝。即子进程从父进程得到数据段和堆、栈段的拷贝,这些需要分配新的内存(不是与父进程共享,而是单独分配内存);而对于只读的代码段,通常使用共享内存的方式访问。
fork返回后,子进程和父进程都从调用fork函数的下一条语句开始执行。
由于子进程与父进程的运行是无关的,所以,父进程可先于子进程运行,子进程也可以先于父进程运行
eg:
myfork.c
wKioL1ePL9DCNzNVAACFfsEw0aY947.jpg
Makefile
wKioL1ePL_bRlFViAAAZrZQKSKM084.jpg

运行结果
wKiom1ePMBfCyuxCAAA64raeeZ8302.jpg
以前的fork创建一个子进程时,将会创建一个新的地址空间,并且拷贝父进程的资源,然后将会有两种行为:1.执行从父进程那里拷贝过来的代码段(进程希望复制自身,从而父子进程能同时执行不同段的代码);2. 调用exec执行一个新的代码段(进程想执行另外一个程序)
当进程调用exec时,一个进程替换了当前进程的文本、数据、栈、堆段。这样,前面的拷贝工作就白费力气了,这种情况下,人们想出了vfork。
vfork并不复制父进程的进程环境,子进程在父进程的地址空间中运行,所以子进程不能进行写操作,并且儿子“霸占”着父亲的房子的时候,就要委屈父亲一下,让他在外面歇着(阻塞),一旦儿子执行了exec或者exit后,相当于儿子买了属于自己的房子,这时候就相当于分家了
2.vfork()函数
wKiom1ePMIzgEAaVAAAoV9Kz-TI441.jpg
vfork创建新进程的主要目的在于调用exec函数执行另外的一个新程序,在没调用exec或exit之前,子进程的运行是与父进程共享数据段的。
vfork调用中,子进程先运行,父进程挂起,直到子进程调用exec或者exit,在这以后,父子进程的执行顺序不再被限制。
eg:
myvfork.c
wKioL1ePMLehki8JAACBKtfjyKo084.jpg
Makefile
wKioL1ePMNmQxfe4AAAbxuOhsUM358.jpg

运行结果
wKioL1ePMQWh0QwcAAAuxVr3icY081.jpg

一个经典的例子
test.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <unistd.h>
void test()
{
    pid_t pid;
    pid=vfork();
    if(pid<0)  //失败
    {
        printf("vfork error\n");
        exit(0);
    }
    else if(pid==0)
    {
        printf("1:child pid=%d,ppid=%d\n",getpid(),getppid());
        return;  //return执行完后,把控制权交给调用函数,而exit()执行完后把控制权交给系统 ,改成_exit(0),则会有另一种结果
    }
    else
    {
        printf("2:parent pid=%d,ppid=%d\n",getpid(),getppid());
    }
}
void fun()
{
    int i=0;
    int buf[100];
    for(;i<100;i++)
    {
        buf=0;
    }
    printf("3:child pid=%d,ppid=%d\n",getpid(),getppid());
}
int main()
{
    test();
    fun();
    return 0;
}



Makefile
wKiom1ePMWbQrdpWAAAVUET55ZY671.jpg
运行结果
wKiom1ePMYWj1AUrAAA0ClronaU917.jpg

程序在后续执行时出现了错误,并且可知道是在父进程中出现的错误
vfork函数调用时,子进程比父进程先运行,在调用test()函数执行时,子进程执行完之后,将清理test函数的栈空间,然后子进程再调用fun()函数,将覆盖掉test的栈空间,继续执行fun函数。但是,当子进程退出后,执行父进程,但是,在test函数返回的时候该栈空间已经被子进程破坏了,不存在了,所以就出现了栈错误
区别:
1.vfork保证子进程先运行,在它调用exec或者exit之后,父进程才可能被调度运行,之后,父子进程的执行顺序才不再有限制。如果在调用exec或者exit之前,子进程依赖于父进程的进一步动作,则会导致死锁
2.fork要拷贝父进程的进程环境(数据段),而vfork不需要完全拷贝父进程的进程环境(数据段),在调用exec或者exit之前,父子进程共享进程环境(数据段),相当于线程概念,此时父进程阻塞等待(因为子进程先运行)。
结束子进程:
exit和_exit函数用于正常终止一个程序,_exit()立即进入内核,exit()则需要先执行一些清除处理(包括调用执行各终止处理程序,关闭所有标准I/O流等),然后再进入内核。
结束子进程不用exit(0),而使用_exit(0)。因为_exit(0)在结束进程时,不对标准I/O流进行任何操作,而exit(0)回关闭进程的所有标准I/O流



运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-247129-1-1.html 上篇帖子: CENTOS 7 如何修改IP地址为静态 下篇帖子: CentOS6.5下调用外部邮箱发送邮件 linux Linux
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表