前言:
前面我们已经学习了进程初识,现在我们来深入进程的学习,也就是进程的控制
1.前面我们已经了解了fork创建子进程,但是这篇博客我们将结合前面的虚拟地址空间与物理地址空间,再次深入学习fork函数
在 linux 中 fork 函数是⾮常重要的函数,它从已存在进程中创建⼀个新进程。新进程为⼦进程,
⽽原进程为⽗进程
我们创建一个code.c来实验一下
cpp
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
int main()
{
pid_t id = fork();
if (id == 0)
{
printf("我是子进程,pid : %d, ppid : %d\n", getpid(), getppid());
}
else
{
printf("我是父进程,pid : %d\n", getpid());
}
return 0;
}
运行一下就有如下结果:

返回值:⼦进程中返回0,⽗进程返回⼦进程id,出错返回-1
1. 为什么要给⼦进程返回0,⽗进程返回⼦进程pid?
因为父进程有很多子进程,所以需要每个子进程的pid,而子进程只有一个父进程,可以通过ppid来找到,所以只需要.-1 / 0来判断是否创建成功
2. 为甚⼀个函数fork会有两个返回值?
因为有两个进程在进行代码
3. 为什么⼀个id即等于0,⼜⼤于0?
这就是虚拟地址的原因,两个进程的id指向同一块虚拟地址,但是不同进程通过该虚拟地址映射到的物理地址不一样,导致id即等于0,⼜⼤于0
进程调⽤ fork ,当控制转移到内核中的 fork 代码后,内核做:
• 分配新的内存块和内核数据结构给⼦进程
• 将⽗进程部分数据结构内容拷⻉⾄⼦进程
• 添加⼦进程到系统进程列表当中
• fork 返回,开始调度器调度
可以用下图来表示fork的过程

所以,fork之前⽗进程独⽴执⾏,fork之后,⽗⼦两个执⾏流分别执⾏。注意,fork之后,谁先执⾏完全由调度器决定
2.对于上面的问题三,我们刚刚只是简单的解答,下面我们来具体解答
写时拷⻉
通常,⽗⼦代码共享,⽗⼦再不写⼊时,数据也是共享的,当任意⼀⽅试图写⼊,便以写时拷⻉的⽅式各⾃⼀份副本。具体⻅下图:

因为有写时拷⻉技术的存在,所以⽗⼦进程得以彻底分离离!完成了进程独⽴性的技术保证!
写时拷⻉,是⼀种延时申请技术,可以提⾼整机内存的使⽤率
所以刚刚发生了写实拷贝,使id = 0 且 > 0!!
3.fork常规⽤法
• ⼀个⽗进程希望复制⾃⼰,使⽗⼦进程同时执⾏不同的代码段。例如,⽗进程等待客⼾端请求,
⽣成⼦进程来处理请求
• ⼀个进程要执⾏⼀个不同的程序。例如⼦进程从fork返回后,调⽤exec函数
4.fork调⽤失败的原因
• 系统中有太多的进程
• 实际⽤⼾的进程数超过了限制