- 上一篇博客我说到了 函数 fork,所以我只是说一下算法。
函数 vfork
-
vfork 函数调用序列和返回值与fork相同,但两者的语义不同。
- vfork 起源于较早的2.9 BSD。但是BSD的开发者在4.4BSD中删除了这个函数,但是4.4BSD派生的所有源码BSD版本又将其收回。在SUBv3中,vfork 被标记为弃用的接口,在SUSv4中被完全删除。因此可移植的应用程序不应该使用这个函数。
-
vfork和fork一样会创建一个子进程,但是它不将父进程的地址空间完全复制到子进程中,因为子进程会立即调用exec(或者exit),于是也就不会引用该地址空间。不过在子进程调用exec或者和exit之前,它在父进程的空间中运行。
-
上面的这种方式的问题:这种优化方式在某些系统的实现中提高了效率,但是如果子进程修改了数据、进行函数调用或者没有调用exec或exit就会返回,都可能会带来未知的结果。
-
另一个区别:vfork保证子进程先运行,在他调用exec或者exit之后父进程才能被调度运行,当子进程调用这个两个函数的任意一个时,父进程会恢复运行。
-
注意:如果在调用这两个函数之前子进程依赖父进程的进一步动作,则会导致死锁。
1 #include<unistd.h>
2 #include<stdio.h>
3 #include<stdlib.h>
4
5 int globvar = 6;
6
7 int main()
8 {
9 int var;
10 int pid;
11
12 var = 88;
13 printf("before vfork\n");
14 pid = vfork();
15 if(pid < 0)
16 {
17 perror("vfork error");
18 }else if(pid == 0)
19 {
20 globvar++;
21 var++;
22 exit(0);
23 }else
24 {
25 printf("pid = %d, glob = %d, var = %d\n",getpid(), globvar, var);
26 exit(0);
27 }
28
29 return 0;
30 }
- 我们看到子进程对变量做出增1的操作,结果改变了父进程中的变量值。因为子进程在父进程的地址空间中运行,所以这并不令人惊讶。但是其作用的确与fork不同。