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并没有改变。