fork函数
使用fork需要包含<unistd.h>
函数原型:
pid_t fork(void)
返回值:
返回值有两个 父进程返回子进程的pid,子进程返回0,如果进程创建失败返回-1
fork()的运行规则
1、fork()在创建子进程的过程中,会将父进程的PCB拷贝一份,同时修改PCB中的pid 以及 ppid
2、fork()会会把父进程的虚拟地址空间拷贝一份,从而作为子进程的虚拟空间。
这里的拷贝可不是单纯的浅拷贝,而是写时拷贝,使得父子进程共用同一段代码,但却各自有自己的一份数据
3、使用fork创建子进程时,父进程与子进程执行代码的顺序没有先后,完全依赖于调度器的调度。
接下来,我们来分析一段代码,从而更清楚的了解fork()的运行过程。
#include <stdio.h>
#include <unistd.h>
int main()
{
int i;
for(i=0;i<2;i++)
{
fork();
printf("=");
}
return 0;
}
这个程序输出是什么结果呢???
结果是八个等号。
具体讨论:
1、首先程序只有一个父进程,在i=0时进入循环,调用fork()产生第一个子进程。
2、子进程创建成功后,父子进程执行同一段代码" printf("=") ",此时父子进程中的缓冲区都有一个'='
3、i++,后进入第二层循环,父进程再次调用fork()产生子进程,第一个子进程也调用fork()产生子进程,此时公有4个进程
4、因为在调用fork()时会拷贝父进程的虚拟地址空间,所以第二次产生的两个子进程中的缓冲区都有一个'=',并且此时i=1
5、紧接着4个进程同时执行代码" printf("=") ",所以缓冲区一共有8个'='
在清楚了解上述代码执行的情况后,我们再来分析一段代码
#include<stdio.h>
#include<unistd.h>
int main()
{
int i;
for(i=0;i<2;i++)
{
fork();
printf("=");
fflush(stdout);
}
return 0;
}
这时这段代码输出的又是什么呢?
此时结果是:六个等号
具体讨论:
1、首先程序只有一个父进程,在i=0时进入循环,调用fork()产生第一个子进程。
2、子进程创建成功后,父子进程执行同一段代码" printf("=") ",此时父子进程中的缓冲区都有一个'='
3、在执行完" printf("=") "后,执行的是刷新缓冲区的操作,故此时父子进程中的缓冲区都是空的,此时控制台已经输出了两个'='
4、i++,后进入第二层循环,父进程再次调用fork()产生子进程,第一个子进程也调用fork()产生子进程,此时公有4个进程
4、因为在调用fork()时会拷贝父进程的虚拟地址空间,所以第二次产生的两个子进程中的缓冲区都是空的,并且此时i=1
5、紧接着4个进程同时执行代码" printf("=") ",所以缓冲区一共有6个'='