Linux之fork与vfork区别

创建一个新进程的方法只有由某个已存在的进程调用fork()或vfork()

Linux之fork与vfork区别



1.fork()函数

Linux之fork与vfork区别




返回值:成功:父进程:返回子进程的PID
子进程:返回0
失败:父进程返回-1
子进程是父进程的一个拷贝。即子进程从父进程得到数据段和堆、栈段的拷贝,这些需要分配新的内存(不是与父进程共享,而是单独分配内存);而对于只读的代码段,通常使用共享内存的方式访问。
fork返回后,子进程和父进程都从调用fork函数的下一条语句开始执行。
由于子进程与父进程的运行是无关的,所以,父进程可先于子进程运行,子进程也可以先于父进程运行
eg:
myfork.c

Linux之fork与vfork区别



Makefile

Linux之fork与vfork区别




运行结果

Linux之fork与vfork区别




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

Linux之fork与vfork区别




vfork创建新进程的主要目的在于调用exec函数执行另外的一个新程序,在没调用exec或exit之前,子进程的运行是与父进程共享数据段的。
vfork调用中,子进程先运行,父进程挂起,直到子进程调用exec或者exit,在这以后,父子进程的执行顺序不再被限制。
eg:
myvfork.c

Linux之fork与vfork区别



Makefile

Linux之fork与vfork区别




运行结果

Linux之fork与vfork区别




一个经典的例子
test.c

#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[i]=0; 

    } 

    printf("3:child pid=%d,ppid=%d\n",getpid(),getppid()); 

int main() 

    test(); 

    fun(); 

    return 0; 

内容版权声明:除非注明,否则皆为本站原创文章。

转载注明出处:https://www.heiqu.com/15142.html