您的位置:首页 > 其它

vfork()

2016-05-03 11:05 239 查看
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并没有改变。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: