vfork()

vfork函数的调用序列和fork函数一样,但两者语义不同。

vfork创建一个进程,而新进程的目的是exec一个新程序,如shell。

vfork与fork区别:

1 fork:子进程拷贝父进程的数据段、堆栈段

 vfork:父子进程共享数据段

2  vfork并不将父进程的地址空间完全复制给子进程,因为子进程会立即调用exec或exit,也就不会访问该地址空间,只在子进程调用exec之前,它在父进程空间中运行。

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

使用fork:

<span style="font-size:18px;">#include<sys/types.h>
#include"ourhdr.h"
int glob=6;
char buf[]="a write to stdout\n";
int main(void){
    int var;
    pid_t pid;
    var=88;
    if(write(STDOUT_FILENO,buf,sizeof(buf)-1)==-1)
        err_sys("write error");
    printf("before fork\n");
    if((pid=fork())==-1) err_sys("fork error");
    else if(pid==0){
        glob++;  
        var++;
    }
    else{
        sleep(2);
    }
    printf("pid=%d,glob=%d,var=%d\n",getpid(),glob,var);
    
}</span>
输出:

a write to stdout
before fork
pid=3155,glob=7,var=89 //子进程变量改变
pid=3154,glob=6,var=88//父进程变量未改变,原因是父子进程不共享地址空间



使用vfork

<span style="font-size:18px;">#include<sys/types.h>
#include"ourhdr.h"
int glob=6;
char buf[]="a write to stdout\n";
int main(void){
    int var;
    pid_t pid;
    var=88;
    if(write(STDOUT_FILENO,buf,sizeof(buf)-1)==-1)
        err_sys("write error");
    printf("before vfork\n");
    if((pid=vfork())==-1) err_sys("vfork error");
    else if(pid==0){
        glob++;  
        var++;
        _exit(0);
    }
    else{
        sleep(2);
    }
    printf("pid=%d,glob=%d,var=%d\n",getpid(),glob,var);
    
}</span>
输出:

a write to stdout
before vfork
pid=3163,glob=7,var=88

vfork只printf一次,因为子进程中调用了_exit(0),由于vfork后父子进程共享数据段,所以父进程中全局变量glob改变了,而堆栈段的var并没有改变。

你可能感兴趣的:(vfork())