vfork函数创建新进程的主要目的在于用exec函数执行另外的程序。在没调用exec或exit之前子进程的运行中是与父进程共享数据段的,在vfork调用中,子进程先运行,父进程挂起,直到子进程调用exec或exit,在这之后,父子进程的执行顺序不在有限制。
例:
1 #include<stdio.h>
2 #include<stdlib.h>
3 #include<unistd.h>
4 #include<sys/types.h>
5
6 int main()
7 {
8 int g_val = 100;
9 pid_t id = vfork();
10 if(id == 0)
11 {
12 g_val++;
13 printf("child,running first...%d:%p\n",g_val,&g_val);
14 sleep(1);
15 //exit(0);
16
17 }
18 else
19 {
20 printf("father,running second...%d:%p\n",g_val,&g_val);
21 }
22 return 0;
23 }
在执行这个程序之后,会发生什么情况呢?运行结果如下:
在打印了第一行child,running first之后,每隔一秒打印两行father...和child...。
这是由于在子进程调用exec或exit之前,在父进程空间中运行,也就是说会更改父进程的数据段,栈和堆。每个进程在内核中都有一个进程控制块(PCB)来维护进程相关的信息。子进程在执行sleep(1)时,更改了父进程中进程控制块的程序计数器,程序中将被执行下一条指令的地址被修改为子进程的地址,所以在父进程结束后又执行了子进程。如此循环往复就得到了我们看到的情况。
若在源文件中加上exit(0),结果如何?
在打印child...之后一秒打印father...程序运行结束。由于子进程在sleep(1)后退出,那么父进程结束后程序结束。
总结:
1:vfork保证子进程先运行,在它调用exec或者exit后父进程才被调度运行;
2:在子进程调用exec或exit之前,在父进程空间中运行,会改变父进程的pcb。