1、fork
fork:Linux下的fork()为复制进程,就是以父进程为模板复制创建子进程。执行fork()函数,会将运行着的程序复制一份,分别称为:父进程和子进程
它是Linux下一种特别的创建子进程的函数,这个函数在执行过程会有两个返回值:
返回值:
- 负值:创建(复制)子进程失败
- 0值:返回到新创建的子进程
- 大于0的值:返回父进程或者调用者,该值包含新创建的子进程的PID
2、用法
pid_t pid = fork(); // 如果创建成功,会有两个进程:父进程和子进程,fork在这两个进程中分别返回,就是返回两次且返回值不同
在操作系统中 ,每一个进程都有对应的一个PCB(进程控制块)
,PCB是用:struct tak_struct这个结构体来表示,由于系统中不止一个进程,会存在很多进程,系统会将这些进程用双向链表
链接起来。如下图所示:
当我们执行pid_t pid = fork();
时,会将当前进程复制一份,其复制过程如下:
- 首先将当前PCB块中的pid块复制一份,此时将pid换成子进程的pid,因为pid不能重复(类比于身份证号),一个pid唯一标识一个进程
- 接着复制进程实体
【注】复制进程,不是一次性全部复制,先复制PID,再接着处理进程实体
【注】
- 注意,程序的执行:当执行fork以后,父进程个子进程会同时执行,
子进程会从fork这个位置以后的代码开始执行
- 父进程中,fork的返回值为:子进程的PID(大于0);子进程中,fork的返回值为0
接下来,看个简单的代码:
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h> // fork的头文件
int main()
{
int n = 0;
char* s = NULL;
pid_t pid = fork();
if(pid == -1)
{
exit(0); // 说明创建失败
}
if(pid == 0)
{
n = 3; // 子进程
s = "child";
}
else
{
n = 7; // 父进程
s = "parent";
}
for(int i = 0; i < n; i++)
{
printf("s = %s\n", s);
sleep(1); // 休眠1s
}
exit(1);
}
分析:因为执行fork,会复制一个子进程,此时父进程和子进程会同时执行,但由于屏幕每次打印只能显示一个,所以不管是显示父进程还是子进程,运行结果均正确,但需要记住的一点就是:两个进程同时执行