fork函数
fork函数会从已经存在的进程中创建出一个新进程。新进程为子进程,而原进程为父进程。
#include <unistd.h>
pid_t fork(void);
//返回值:子进程返回0,父进程返回子进程pid,出错返回-1;
进程调用fork(),当控制转移到内核中的fork代码后,内核做:
- 分配新的内存块和内核数据结构给子进程
- 将父进程部分数据结构内容拷贝至子进程
- 添加子进程到系统进程列表当中
- fork返回,开始调度器调度
代码实例:
#include<stdio.h>
#include<unistd.h>
#include<stdlib.h>
int main(){
pid_t pid;
printf("before:pid is %d\n",getpid());
if((pid = fork()) < 0){
perror("fork!");
return 1;
}
else{
printf("after:pid is %d,fork return %d\n",getpid(),pid);
sleep(1);
}
return 0;
}
运行结果:
从结果我们可以看到有三行输出,一行before,两行after。父进程8082先打印before消息,然后打印after;而子进程8083却只打印了after,并没有创建进程之前的before。这是由于fork函数的特性所决定的,请看以下图解:
【结论】:fork之前父进程独立执行,fork之后,父子进程两个执行流分别执行。fork之后,谁先执行完全由调度器决定。
写时拷贝
通常父子进程代码共享,父子进程在不写入时,数据也是共享的;但是,当任意一方试图写入时,便以写时拷贝的方式各自复制一份副本,如图所示:
fork常规用法:
- 一个父进程希望复制自己,使父子进程同时执行不同的代码段。
- 一个进程要执行一个不同的程序。
fork调用失败:
- 系统中有太多进程。
- 实际用户的进程数超过了限制。