进程 概念和理解 - Linux 是怎么做到 管理进程的?-fork 手动创建进程

news2025/1/14 3:19:39

前言

上一篇博客当中,对 冯诺依曼体系结构 和 操作系统 进行了简要概述,本篇博客将会从上一篇博客的基础之上进行展开,如果你有些不了解的话,建议先看上一篇博客再看本篇博客:

冯诺依曼体结构 - 为什么要有操作系统-CSDN博客

进程概念

 就算你不知道 操作系统,一大概率听说过进程,其实进程就是 已经加载到内存当中程序正在运行的程序) ,进程是一定要被加载到内存的 ,这些程序就是又程序员编写的 一个一个软件。其实 一个进程本质上也可以称之为是一个 任务,这样获取你会有更好的理解。

在windows 当中,按下 ctrl + shift + esc 就可以的打开任务管理器,或者邮件任务栏,也可以打开,当我们打开 任务管理器之后,就可以的看到当前你的计算机正在执行哪一些进程:

 在 linux 当中可以使用 以下命令来查看当前 那些进程正在被执行:
 

ps axj

其中的 PID 就是这个进程专属的 进程编号。 

PPID (parent process ID)就是 当前进程的 父进程 PID

像上述的 ps ajx | head -1 && ps ajx | grep myprocess 这个指令当中,使用了 "&&" 这个符号,这个符号表示要左右两边的指令都运行成功才能输出结果,同样的,在同一位置我们可以使用 ";" 分号来代替他,这个 ";" 分号的意思就是 左右两边的指令都要执行。

我们发现上述多过滤出一个 进程 :

这个其实是 grep 自己的进程,以为 grep 要过滤就要先把自己给 调入内存,自己运行起来,然后才能查找。因为 上述命令当中 grep myprocess 本身就到了 myprocess 这个关键词,所以就把自己给查找出来了。 

其中还有一个 COMMAND 这个属性,代表的是这个进程当前是 使用了哪一个命令调用的进程:

 或者使用 Top 命令来查看当前正在运行的进程信息:

 进程的理解

 首先,进程再被加载到 内存之前,是一个程序,那么一个程序,就又代码文件和一些附带文件所存储,既然是文件,那么一个程序在被加载之前,就是在磁盘上存储的;

也就是说,进程再被加载到 内存之前,实在磁盘上进行存储的,所以,对于 冯诺依曼体系结构当中的 输入设备,对于进程来说,就是磁盘了。

 此时,在 磁盘当中的 程序要想被cpu执行,就要先加载到 内存当中,而 程序文件,本质上也就是存储一些二进制的文件,由 代码 和 数据构成的程序二进制文件。都是二进制的数据,只是有些二进制数据 表示代码,有些表示数据。

那么 ,是代码的二进制数据就交给 控制器去执行;是 数据的二进制数据就交给 运算器去执行。

归根结底还是 把 磁盘的当中的程序的二进制数据 读取到了内存当中。

那么问题来了,虽然cpu 当中一次只能执行一个任务,但是,一个操作系统不可能值运行一个进程,可以同时运行多个进程。比如:我既可以使用 QQ音乐听歌,又可以使用 word 写文档;还可以打开浏览器 看 B站。

但是,在上述说的不同的 进程当中,我可能先写 word 文档,在看 B站,当B站的当前视频在写完 word 之前就播放完了,此时我想就不看 B站视频了,那么 浏览器就被我关掉结束了,但是此时我还是再写 word 文档。

也就是说,各个进程之间是什么时候开始的,什么时候执行结束的,我们不能完全去控制,他们在同一时间片当中,有各自进程的执行状态。那么,在上述情况下,操作系统必须管理好各个进程如何让操作系统合理的运行?就是需要解决的问题。

 如何管理进程?

 在上一篇博客对操作系统的描述当中就说到了,要想让计算机帮我们处理数据,就得前把数据描述起来(用 struct 或者 class 把各个对象的属性包装起来)。

那么接下来的工作就是,管理好由各个进程 包装成的一个一个对象,那么无非就是 把各个进程增删查改,就得用 某种数据结构来对这些个对象,进行某种逻辑上的链接。

 任何一个进程,在加载到内存时,在由 程序形成 真正 的进程之前,要对 这个程序当中 属性 管理这个进程需要的 属性 包装到一个 结构体(或者类,因为操作系统是由C实现的,所以这里统称为结构体)当中,用于描述这个进程。我们把这个结构体对象称之为 -- PCB(process ctrl block)-  进程控制块

 其实计算机来辨别新事物 的方式 和 人是一样的,任何在辨别新事物的时候,其实很难的看到这个事物的本身,我们认识到 这个事物 ,知道这个动物叫做 老虎,是它表现出来的特征,让我们认为他是一个老虎;人是通过 属性 来认识事物的计算机也是一样

 当某个事物表现的属性够多时,这些属性的集合(对象)就是目标对象。

 在描述进程也是一样的,当描述一个进程的属性足够多的时候,那么就可以认为,这是一个进程(可以理解为 进程 就是属性的集合)。这本质上就是一种 面相对象

 在用C实现的操作系统当中,一个属性的集合 其实就是一个 struct(结构体)


 那么,进程之前的属性千奇百怪,操作系统可以创建很多个 结构体,操作系统如何分辨哪一个是哪一个进程的结构体呢?

其实很简单,在学校当中,假设是新生,老师不知道名字,长相等各个属性的情况下,可以如何分辨这些学生呢?

就是学号,新生的名字可能是一个生僻字,老师不会读,各个新生可能有自己的爱好和特长,老师一时半会也分辨不出来。但是,数字老师总得认识吧,那么他只需要从 学校教务处拿到一个新生名单,就可以分按照学号来分辨新生了。

同样,计算机可能不会认识哪一个变量代表的是什么意思,但是,如果给每一个进程都编上属于这个进程唯一的编号,那么计算机总得是可以分辨数字的,所以就可以分辨出进程。

 除了进程编号之外,还会有 优先级,进程的状态等等 共同属性,多少可以帮助 操作系统识别进程的。


总结:

当一个程序被加载到内存当中时,操作系统首先要干的事情就是,根据这个 程序的PCB类型,为这个进程开辟一个 PCB对象,对这个对象当中的各个属性进行初始化。

在上述的过程当中,操作系统可以选择先不加载 程序当中的 数据 和 代码的二进制数据到内存当中,可以先为 这个进程构造一个 PCB 对象。

但是,程序的当中的二进制数据总是要加载到 内存当中的,那一个PCB 当中只是关于这个对象的属性的存储,但是 进程的 二进制数据还是要开空间来存储:就好比是:你养了一只狗,你用笔记录了 这只狗的 属性特征,这个的存储大小是可能是一张纸,或者是一个笔记本;但是,你要想在家当中养活这只狗,那么得为这个狗买狗窝,给他空间作家。


什么是进程(理解 管理进程的过程)

所以,一个 内核PCB数据结构对象 不叫进程;一个 code & data (文件二进制数据(代码和数据)) 也不叫进程,这两个 加起来 才叫一个进程。

操作系统会把 一个 PCB  和 一个 code & data(文件二进制数据) 构建成一个 结构体对象,这个结构体对象是由操作系统自己生成的。

 所以,操作系统要管理这个进程,根本就不看 这个进程的 代码和数据,只看 PCB对象当中的属性就行了。

 我可以使用数据结构把 各个进程当中的 PCB 对戏那个按照这个数据结构的链接方式进行链接,此时,我们要想对这个进程进行管理,就只需要对这个 存储 各个 PCB 对象数据结构的增删查改进行 操作即可。

比如,在我们 PCB对象当中增加一个 PCB* next 指针,指向下一个 PCB对象,那么现在各个PCB 就是一个 单链表了,此时,在操作系统当中,对进程进行管理,变成了对 这个 PCB对象的单链表进行增删查改的操作了。


所以,我们可能会看到某一个进程正在排队等待运行,不是这个进程的 二进制文件数据 在排序,而是这个 进程的 PCB对象正在排队

就像在公司当中投简历排队一样,面试官是按字符串查找它需要的什么技能的人,然后去查找简历当中符合要求的人,此时,如果你等不及了,打电话问公司人力资源部咨询人员,他会告诉你,你正在排队,或者你已经被录取,或者你被淘汰。而不是 公司的 leader 一个一个人打电话去问,这就太挫了。

如果你此时正在排队,是你的 投进去的简历在公司当中的队列当中排队,而不是你这个人在公司当中排队。


总结:一个进程包括 属性的和数据,数据是这个进程运行自己的算法逻辑,和一个需要的数据;属性是 操作系统管理这个 进程。什么时候等待,什么时候运行,什么时候结束(死亡)等等操作所需的 PCB对象当中的属性。

操作系统管理 进程是他通过属性进行管理,而不是通过进程当中的 数据。

Linux 是怎么做到 管理进程的?

各个操作系统之间虽然都是按照 先描述在组织的方式来实现的,但是,对于操作系统的实现细节还是很大差异的。

比如:各个操作系统之间的 关于 PCB 的数据结构的实现就不同。

描述 Linux进程的 - PCB

进程信息 被放在一个叫做 进程控制块数据结构 中,可以理解为 进程属性的集合

课本上称之为PCB(process control block),Linux操作系统下的PCB是: task_struct

这个 task_struct 是一个大型的结构体,这里面包含了Linux 内核当中 描述进程 所以需要用到的属性。 

 首先应该理解的是,task_struct 是 PCB 的一种,属于 PCB。在windows ,macOS 当中都有自己各有的 PCB。

  • Linux描述进程的结构体叫做 task_struct
  • task_struct是Linux内核的一种数据结构,它会被装载到RAM(内存)里并且包含着进程的信息
     

而在Linux 当中 链接 各个 PCB对象(注意:只是PCB对象,而不是 二进制数据 和 PCB对象),最基本组织进程 tash_struct 的方式  ,一般是使用 双向链表来 链接的。

而且,在,对于一个 进程 PCB 对象,不仅仅会放在的一个 双链表当中,还可能放在一个 队列当中。其实想做到这个并不难,在 C语言当中,只需要多创建一个 数据结构的 PCB* 指针就行。

比如有等待队列,运行队列等等数据结构,我们想把某一个进程 当前状态进行修改,就只需要把这个 进程 的PCB 对象放到 某一个 在组织的数据结构 当中就行,至于操作系统怎么运行,是按照这个数据结构当中的算法来执行的。

数据结构背后是配套的算法,而算法背后又是具体的应用场景。

 task_struct 的内容分类

  •  标示符: 描述本进程的唯一标示符,用来区别其他进程。
  • 状态: 任务状态,退出代码,退出信号等。(比如,当前进程执行的状态怎样的,是再等待运行代码?还是正在运行代码?代码运行后它的状态码是什么?进程是新建的,正在运行的,正在休眠的,死亡的等等状态)
  • 优先级: 相对于其他进程的优先级。(操作系统当中是有很多个进程在同时运行的,那么进程和进程之间就存在这竞争关系)
  • 程序计数器: 程序中即将被执行的下一条指令的地址。(程序一般是 在栈帧当中顺序执行的,但是,程序一般情况下不是从上往下顺序跑完的,可能还调用函数,循环等等往返着调用语句,这时,我们如何在在调用完函数,知道下一个应该执行完哪一个语句,就需要 PC指针,或者程序计数器来记录跳转执行指令下一个语句的指针。这个程序计数器是 cpu 的一个寄存器,这个寄存器就用来存储下一个语句的地址)
  • 内存指针: 包括程序代码和进程相关数据的指针,还有和其他进程共享的内存块的指针
  • 上下文数据: 进程执行时处理器的寄存器中的数据[休学例子,要加图CPU,寄存器]。
  • I/O状态信息: 包括显示的I/O请求,分配给进程的I/O设备和被进程使用的文件列表。
  • 记账信息: 可能包括处理器时间总和,使用的时钟数总和,时间限制,记账号等。(比如:某某个进程累计在cpu 当中运行了多少时间)
  • 其他信息

在 Linux 当中有一个 目录(/proc):(process 的简写),当中就放了 现在执行的 动态运行的 所以进程 信息:

 我们发现,此时,由很多蓝色的数字目录,正在运行:

而且,这些目录是以数字命名的,这个数字其实就是 某一个进程的 PID。所以,我们就理解了,如果某一个进程开始运行了,那么 操作系统就会在 /proc 这个目录当中,以这个进程的 PID 创建一个 目录,这个目录当中保存了这个进程的大部分属性

当这个进程结束执行结束之后,在 /proc 目录当中创建的 以 PID 为名字的文件夹就会被自动销毁。 

我们现在可以查看某一个进程当中的 各个属性的文件:
 

 其中有一个 exe 文件夹,这个文件当中指向的就是 这个可执行文件的 绝对目录地址。

还有一个 cwd,当前进程的工作目录。比如:我们使用 C当中的 fopen()函数,自动帮我们创建一个文件的时候,是在当前目录下创建的,而这个 cwd 所指的目录就是这里所指的当前目录。

比如现在使用 touch 命令创建文件的时候,意思就是在当前目录下创建文件,你说,这不是废话吗?但是,touch指令是写的一个软件实现的,那么touch 指令要运行,就要在提取到内存当中进行运行,这时就需要 cwd了。所以,cwd的意思就是:当前进程是在哪一个目录下运行的,那么 cwd 就指向哪一个目录。


我们可以直接使用  kill -9 进程编号 命令 ,来直接干掉一个进程,这个命令的意思就是给 这个进程发送 9 号信号:

kill -9 xxxxx

 有时,进程可能会 奔溃,卡死,等等情况,我们可以使用 ctrl + c 来结束,但是有时候 ctrl + C 都不能结束,就可以使用 上述指令来强制结束一个进程。


在上篇博客对 为什么要有操作系统当中我们而已提到了,操作系统在链接 各个 进程的过程中,本质是对 各个进程的 PCB对象进行 链接,而链接方式可能是各种的数据结构算法,比如:在Linux 当中就是以 双链表的方式来连接各个 PCB 对象的。

那么 像我们在开始使用的 类似 ps axu 指令,查看 当前运行的各个进程的指令。本质上就是用 C/C++ 写的 遍历在操作系统当中 存储链接各个 PCB 的数据结构

 所以,如果现在用户想要访问到当前某个进程的 PID 进程编号的话,不期望直接使用 C 中 访问结构体的方式来访问到 某一个结构体对象当中的 成员属性。而是像C++ 当中访问类的私有成员一样,给出一个接口来访问。

所以,我们在系统上想访问到 某一个 进程的信息,就要使用 类似 ps 这样的指令,也就是要用要用 系统调用接口 的方式来调用。


 我们可以用 一个简单的 命令来实时监控某一个进程(就是用命令当中的一个 一直执行循环来执行 )

# 下述 grep 当中的 -v 选项是反向选择
while :; do ps axj | head -1 && ps axj | grep text | grep -v grep; echo "--------------------------------" sleep 3; done


此时他就会实时监控 text 这个程序是否成为一个进程,像上述,因为text可执行程序没有执行,所以,上述什么都没有检测到,当我们运行了 text 之后,就可以检测到了,我们在 text 可执行程序当中,使用 getpid()这个函数来获取到 本程序生成的 进程的 PID,这个函数的返回值是 pid_t 是有符号整形:
 

 当 text 进程运行时,就会一直打印 本进程的 PID:
 

此时我们采用上述的检测 进程的 命令再次执行:

先执行的 监视命令。然后在执行 text 进程,发现原本没有的 进程信息就出现了。 


而且,我们发现,同一个 程序,每一次运行操作系统自动生成的 PID 可能是不一样的,但是这是相当正常的。

我们需要注意的是:PID 只在当前进程 允许的过程当中是有效的,如果是这个进程被重新执行了,或者是已经结束执行了,这个原本这个进程的 PID 就不在适用了。

如上所示,两次执行的text程序生成的 PID 都是不一样的 。 


同样的,你还可以使用 getppid()这个函数来获取到 当前程序变成进程 系统生成的 PPID:

 执行结果:

 但是发现,不管怎么运行text这个程序,这个 由 text 生成的进程  的  PPID 都是不变的:
 

 此时我们就查看一下 此时这个 28785 到底是什么?

ps axj | head -1 && ps axj | grep 28785 | grep -v grep

打印:
 

 我们发现 只有一个 PID 为 28785 的进程。

相信你已经猜到了,这个进程就是 bash -- 命令行解释器,这个解释器就可以 解释我们输入的命令从而指令对应命令的软件。

我们在 Linux 当中所有执行的指令都是 由经过命令行解释器的,所有的  命令都是 bash 的子进程。所以,当其中的某一个子进程奔溃了,或者是报错 了,执行错误,都不会影响到 bash 这个进程。只会影响到 子进程。

fork 手动创建进程

 我们之前都是使用类似于 "./" 的方式,直接调用某一个 可执行程序,这样的话,操作系统就会自动帮我们创建一个 管理这个进程的 PCB文件,然后再在 /proc 这个目录之下创建一个 和这个 进程的编号(PID)相同的 文件夹目录, 这个目录当中就存放了 关于这个 进程的属性。

但是上述都是操作系统 自己帮助我们创建的,如果想要手动创建一个进程的话,可以使用 fork

这个是一个函数,我们先使用 man forc 使用手册来查看关于 fork 的介绍
 

fork 是一个函数用于 创建一个子进程。 

此时我们在 Linux 当中编写一个 小程序来验证:
 

在这个进程当中, "end:~!" 在最后只打印了一遍,但是输出却输出了两遍:
 

 在前面两个打印的 begin: this is a process! 和 第一个end:~! 是原本第一个进程执行的结果,其实在 fork()函数之后,后面的代码语句已经被新生成了一个进程,所以,当我们运行 text 这个可执行文件的时候,相当于是运行了 两个进程,后一个进程就是 printf("end:~!\n"); 这个语句

forc()函数的作用就是在 调用的进程为模版,新创建一个 进程。这个新进程就是调用 fork()函数的这个进程 的 子进程。

  fork()函数的返回值是 pid_t 类型:

  • 如果 fork()函数创建进程成功了,那么将会给父进程返回一个当前新产生的子进程的 PID,同时返回 0 给子进程。
  • 如果 fork()函数创建进程失败了,返回 -1 给父进程,此时没有 子进程被创建,给出error 错误码。

相信你已经发现了,当 fork()函数创建进程成功之后,我们看上去返回了两个返回值,一个给 父进程,一个给子进程。

要探究上面的问题,我们下来看这个例子:我们直接在 fork()函数之后,写一个 if 判断一个 fork()函数的返回值。那么此时,如果fork()返回的是 0 ,说明当前执行的是父进程,如果返回的是一个 >0 的数,那么说明当前进行的是 子进程,如果返回的是 <0 的数,返回的是一个 error:
 

 如上所示,为了方便观察,在三种情况之下都 sleep()一下。

输出:
 

 首先,在进程卡开始执行之后执行的是子进程,然执行的是父进程,而且,子进程父进程 之间是交替执行的。但是我们写的 不是 子进程 父进程 分开的两个死循环吗?怎么会 两个交替执行呢?怎么会出现 两个死循环同时再跑呢?

 而且,上述的输出结果是不是就证明了,我们用 Mypid 这个变量来接收的 fork()函数的返回值,等于0 和 大于0 两个条件同时成立了

 这就是因为我们使用了 fork()函数,在 fork()函数之后,变成了两个进程,一个是原本的进程,我们此时称之为 父进程;另一个是 fork()函数新 生成的 子进程。这两个是实实在在的父子关系:

此时你就要注意区分了:

  • 使用 "./" 方式是在指令层面 帮我们创建进程。
  • 使用 fork()函数是在 代码层面 帮我们创建进程。


 我们现在来梳理一下 关于此时 text 这个程序执行的顺序

刚开始执行 text 进程的时候,操作系统就给这个 text 分配了一个 PID 为  18375,此时这个进程也就是 父进程。

在 执行 fork()函数之后,就有 fork()函数一分为二,形成两个分支,一个是之前的进程 --- 父进程;另一个就是 --- 子进程。

之前我们单独 执行 text 进程,就是一个单进程的当时执行的,但是,因为这个 text 当中有一个 fork()函数,这个函数帮助我们创建一个子进程,此时就是多进程的执行方式了。

为什么fork()函数要给 子进程返回 0 ,给父进程返回 子进程 的 PID?

我们先来看,为什么 给 子进程 和 父进程 的返回值要不同?

  •  首先,返回不同的返回值,是为了区分,让不同的执行流,执行不同的代码块。
  • 其次,fork()之后的代码父子共享。像之前给的死循环那个例子,其实 fork()之后的代码,父进程 和 子进程 都是一样的,只不过我们在上述以 fork()函数返回值进行 区分而已。

上述只是解决了 为什么返回值要不同,但是为什么 给子进程返回的是 0 ,而 给父进程 返回的是 子进程的 PID?

 在现实当中,一个父亲可能以后多个孩子,但是一个孩子只能有一个亲生父亲;父进程 可能要对 子进程做一些控制。所以,父进程要拿到 子进程的 PID,用来管理子进程,标识子进程的唯一性;子进程不一样,他只有一个父亲,他只需要找到 PPID 就可以找到父进程

fork()函数干了什么事?

 在 fork()函数 创建新 子进程之前,在系统当中就已经 创建了原本的进程 --- 父进程了,之前说过,进程的本质实际上就是 PCB(内核数据结构) + 代码和数据

在调用 fork()函数之后,系统当中创建一个子进程,创建一个子进程本质上不就是 在系统当中多一个进程吗?

所以,此时在  父进程的PCB 创建的基础之上,由多了一个 子进程 PCB;子进程当中的各个属性其实是按照父进程 为 模版来创建的;

但是此时有一个问题,父进程有 自己的 PCB 和 代码数据,但是子进程是系统创建的,在创建之前没有 像 父进程一样从 磁盘当中获取到 代码和数据,所以,子进程只能无奈的 和父进程 一样,调用同样的代码。

 所以,此处需要注意的是,fork()之后,父子进程访问的代码是共享的。在 C/C++ 当中,代码被加载到内存之后,代码是不能被修改的,能改的只能是代码当中的数据,

为什么要创建子进程呢?如果 子进程和 父进程 执行的是一样的内容,那么直接让 父进程去干不就可以了,为什么要  子进程呢?所以,我们就是要让 父 和子进程 执行不同的事情!像办法让 子进程 和 父进程 执行不同的代码块。这样,让父子做不同的事情,让 父子协同起来。

所以 fork()函数才有了不同的返回值。

 fork()函数是如何做到返回两个返回值的?

我们还是那上述的例子来所说明。

我们需要注意的是:fork()也是一个函数,那么既然是一个函数,那么这个 fork()就具有自己的函数体,也就是 fork()函数自己的代码块。

此时就有个一个问题了,当一个函数执行到 return 语句的时候,这个函数的主要功能有没有实现呢? 在 fork()函数当中是已经做完了的。

在fork()函数的当中会做很多事情,但是根本上都是在创建一个子进程,这里面 最主要的就是 创建和拷贝数据 PCB,给PCB 的属性值初始化。当 父 子 都有了各自的 PCB 之后,CPU 就可以调度这两个进程了。

但是我们刚刚说了,执行到 return语句之后,fork()函数的主要功能已经实现完了;而且 子进程 和 父进程是共用一个代码,但是return 语句是代码吗?答案肯定是的 。所以,也就意味着 return语句也是父子共享的。

所以,当子进程调度 fork()函数时候,就return返回一个值;父进程调度 fork()函数时候也返回一个值。所以这个 fork()函数就被返回了两次。 因为,当你fork ()函数准备return之前,关于子进程的调度早就执行完了,此时,子进程也允许被CPU调度了。所以,再往后,这个 return语句就是被两个 进程所调用,因为 return 语句是代码,被父子进程所共有,当父子进程运行时,就会调用两次return语句。从而实现返回不同的值。

上述的返回值 Mypid 是如何接受两个返回值的?

在上述描述过,父进程因为 在 使用 "./" 调用父进程的时候。就会带上父进程的 数据和代码,所以,父进程是天生就有数据和代码的,但是,子进程却不是,子进程是在父进程当中在创建的,才创建之时,子进程是没有自己的 数据和代码 的,子进程和父进程共用一个代码。

那么问题来了,子进程的数据从何而来,Mypid 变量用于接收 fork()函数的返回值,但是这个 Mypid 变量只有一个啊。子进程运行也是需要数据和代码的。


所以,如果你有上述的疑问,那么你应该首先明确一点:

在任何平台,进程在运行之时,是具有独立性的

 什么是独立性呢?

比如在 windows 当中创建了 QQ 这个进程 和 微信这个进程,如果其中 QQ 这个进程崩溃了,他不会影响到 微信这个进程;如果 微信崩溃了,不会影响到 QQ 这个进程;这两者之间是没有耦合关系的,也就是我们说的 独立性。

独立的本质是一种割裂关系,各个独立的进程,各自就能做自己的事情,自己就能运行(除非他某项功能需要和其他进程来联系)。


所以,上述 父进程 和 子进程都是 一个独立的进程,那么他们就可以自主运行,不需要 相互之间的依靠。既然是自主运行,因为数据可能被修改,那么两者之间肯定是不能用 相同的数据的,肯定是独立的数据。


读到这里,需要区分的是,代码是共享的,因为代码是不能被进程所修改的;而 数据不是共享的,是独立的,因为数据可能会被修改,肯定不能让父子进程访问同一份数据


子进程要想独立一份数据,就需要拷贝一份数据,拷贝的数据是和 父进程共有的数据,拷贝一份给自己用。除此之外,子进程还有各自 专属数据,是父进程没有的,那么在这个份数据当中也应该创建出来。(这个,为子进程拷贝和创建数据工作是由操作系统完成的。

此时,子进程和父进程就拥有了各自独立的数据,不再冲突。

而在访问不同数据,就是通过 父子进程各自的PCB来完成,调用不同的PCB,就访问各自的进程,不管这个代码是不是共享的,但是只要数据不是共享的就行。


那么,上述也提到了 子进程会拷贝 父进程当中数据;但是有一种情况,拷贝下来的父进程数据,对于子进程当中,大部分数据都是没用的。或者说是,拷贝下来的数据有些是子进程使用的。

所以,操作系统在此处进行了优化

首先,在创建子进程之时,不以上上来就拷贝一份父进程的数据,而是,先让子进程和父进程共享父进程的数据。当子进程像访问的时候,如果子进程要对这个数据进行修改,那么就在内存当中重新开辟一块空间,用于存储子进程要修改的变量的值。

这时,操作系统就告诉子进程,你要修改变量数据,不要在父进程的数据区当中直接修改这个变量的数据;而是,在我给你新开辟的一块空间当中修改这个变量的数据。

如此,往后,只要是子进程想要修改 父进程 数据区当中数据,操作系统就会给 子进程在内存当中新开辟一块空间,用于存储要修改的变量的值。

我们把上述的过程称之为 --父子进程之间的  数据层面的写时拷贝

 在用的时候,在给你开辟一块新的空间,不用就不开辟。


此时我们再来理解上述代码:

 fork()函数由父进程调用返回的值,就直接赋值给 父进程的 Mypid变量;如果是子进程调用 fork()函数,那么此时就会发生写时拷贝;相当于,在此时,有两个Mypid 变量,一个是子进程的,一个是父进程。让父子进程在访问 Mypid 这个变量之时,看到是不一样的值。

父进程在访问 Mypid 变量之时,访问的是父进程数据区当中 Mypid 的值;而 子进程在访问 Mypid 之时访问的是 操作系统写时拷贝 的新空间当中的数据。


如果父子进程被创建好之后,fork()往后,谁先运行呢?

 首先,当我们开始运行一个程序,那么这个进程什么时候开始运行我是不能管理的。比如:在windows当中打开了 QQ,那么我们只是告诉了操作系统,此时要运行 QQ了。但是实际上QQ这个进程什么时候开始,是由操作系统决定的。

因为用户只需要使用这个软件即可,什么时候运行进程是由操作系统自己处理的。

所以,谁先运行是由 调度器决定的。

调度器的工作:

操作系统管理进程是管理 用数据结构存储的各个进程的PCB,那么此时,如果用户想运行某一进程,可能就会把这个进程放到 就绪队列当中(打比方),因为在此之前,可能还有很多的进程在排队。那么此时CPU应该挑选哪一个进程开始执行,这个工作就是有 调度器决定

 因为 cpu 在计算机当中是少量的资源,而进程在运行之时都是要 跑到 cpu 之上去运行的,所以,如何更好的利用好 cpu,让我们感受到 很多个进程在同时被调度,调度器 起了很大作用。

 理解 bash 解析命令的过程

在上述理解了 fork()函数之后,我们来了解一下 bash 。bash 是命令解释器,所以的命令都是要通过 bash 解释来运行的。

像上述的使用的 fork()创建子进程的例子,其中的父进程本身就是一个子进程:

我们查看这个 进程pid:

发现其父进程是 bash 。 

所以,bash 在解析命令,然后运行对应进程的过程当中,为了这个进程在运行之后,不用影响到我 bash 的运行,解释其他命令。所以在 bash 当中也是采用 fork()函数来创建子进程的方式来,创建 bash 命令解析的,运行的子进程。

也就是说:bash 在解析命令之后,通过调用 fork()函数创建子进程;至此会后,bash 就会继续进行解释命令的作用,而 bash 创建的子进程就会去执行 bash 解释出的命令的进程。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1127590.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

ResNet论文精读,代码实现与拓展知识

文章目录 ResNet 论文精读代码实现网络可视化代码 拓展知识 ResNets残差的调参残差链接的渊源残差链接有效性的解释ResNet 深度ResNeXt代码实现 能够提点的技巧「Warmup」「Label-smoothing」「Random image cropping and patching」「Knowledge Distiallation」「Cutout」「Ra…

学习如何在linux服务器上修改默认端口22

学习如何在linux服务器上修改默认端口22 修改默认的22端口号重启ssh服务测试连接 修改默认的22端口号 [rootqipa250 ssh]# vim /etc/ssh/sshd_config 知道Port&#xff0c;改为自己想要的端口号 重启ssh服务 [rootqipa250 ssh]# systemctl restart sshd.service测试连接 阿…

Netty使用和常用组件

简述 netty 版本 <dependency><groupId>io.netty</groupId><artifactId>netty-all</artifactId<version>4.1.42.Final </version><scope>compile</scope> </dependency>Netty5 已经停止开发了。 netty 优势 API …

SQL注入专项整理(持续更新中)

深入了解SQL注入 什么是SQL注入&#xff1f; SQL注入即是指web应用程序对用户输入数据的合法性没有判断或过滤不严&#xff0c;攻击者可以在web应用程序中事先定义好的查询语句的结尾上添加额外的SQL语句&#xff0c;在管理员不知情的情况下实现非法操作&#xff0c;以此来实现…

【LeetCode:1155. 掷骰子等于目标和的方法数 | 递归->缓存->dp】

&#x1f680; 算法题 &#x1f680; &#x1f332; 算法刷题专栏 | 面试必备算法 | 面试高频算法 &#x1f340; &#x1f332; 越难的东西,越要努力坚持&#xff0c;因为它具有很高的价值&#xff0c;算法就是这样✨ &#x1f332; 作者简介&#xff1a;硕风和炜&#xff0c;…

查网站域名历史,查域名有没有灰记录,查域名有多少外链的好工具

作为一位建站达人&#xff0c;我一直在寻找高效的网络查询和管理工具。桔子是我近期发现的两款非常实用的浏览器和网站排名查询工具。它们不仅可以帮助我更好地了解域名的外链情况&#xff0c;还支持建立米表和进行网站排名查询。下面&#xff0c;我将详细介绍如何使用这款工具…

【clickhouse】一个性能问题,把一个中间件从头到位翻了个遍

现象 团队在前期引入了clickhouse&#xff0c;业务上端口进行审计&#xff0c;每天的数据量很大&#xff0c;测试小伙伴在测试的时候&#xff0c;使用pcap回放&#xff0c;将千兆带宽几乎跑满&#xff0c;出现每天大概一亿的数据量&#xff0c;导致界面实时查询很慢&#xff0…

分享42个ASP.NET源码总有一个是你想要的

分享42个ASP.NET源码总有一个是你想要的 链接&#xff1a;https://pan.baidu.com/s/1pLbe0hci2PX3IjtxGTDhMw?pwd8888 提取码&#xff1a;8888 项目名称 Asp.Net Core Api项目集成Azure AD实现认证授权 ASP.NET Core 帖子项目 asp.net core 微服务项目 ASP.NET Core 项目…

Spring Security认证源码解析(示意图)

建议先看完Spring Security总体架构介绍和Spring Security认证架构介绍&#xff0c;然后从FilterChainProxy的doFilterInternal函数开始&#xff0c;配合文章进行debug以理解Spring Security认证源码的执行流程。 在之前的Spring Security认证架构介绍中&#xff0c;我们已经知…

C算法:不使用第三变量,实现两数交换

写一个函数实现两数交换&#xff0c;要求不使用第三个变量。 输入样例&#xff1a; 14 16 输出样例&#xff1a; 16 14 代码实现&#xff1a; #include<stdio.h>int main() {int a,b;printf("please input two num:\n"); scanf("%d%d",&…

C# 使用waveIn实现声音采集

文章目录 前言一、需要的对象及方法二、整体流程三、关键实现1、使用Thread开启线程2、TaskCompletionSource实现异步3、将指针封装为Stream 四、完整代码1.接口2.具体实现 五、使用示例方式一方式二 总结 前言 之前实现了《C 使用waveIn实现声音采集》&#xff0c;后来C#项目…

Ubuntu 命令行设置静态IP地址方法

一、先ifconfig查看电脑的网卡信息 找到有线网络或WiFi网络的网卡名称&#xff0c;我这里是eno1 二、输入route -n命令&#xff0c;打印路由表&#xff0c;这里主要是为了查看网关地址 我这里网关地址是192.168.10.1 三、更改配置文件 输入 vim /etc/network/interfaces&am…

企业一般纳税人查询API:简化税务信息获取的利器

前言 随着数字化时代的到来&#xff0c;企业纳税和财务管理领域也经历了革命性的变化。税务管理不再是繁琐的手动工作&#xff0c;而是通过技术工具实现高效和精确。其中&#xff0c;企业一般纳税人查询API成为了企业税务信息获取的强大利器。这一工具不仅简化了税务信息的访问…

微信公众号怎么添加送餐外卖系统

在当今快节奏的生活中&#xff0c;外卖已经成为了人们解决日常饮食需求的重要方式。微信公众号作为一个拥有广泛用户群体的平台&#xff0c;加入送餐外卖系统可以为公众号持有者带来更多的商业机会和用户便利。本文将介绍如何在微信公众号中添加送餐外卖系统&#xff0c;提升公…

系统设计 - 我们如何通俗的理解那些技术的运行原理 - 第七部分:Git、云服务、生产力工具

本心、输入输出、结果 文章目录 系统设计 - 我们如何通俗的理解那些技术的运行原理 - 第七部分&#xff1a;Git、云服务、生产力工具前言Git &#xff1a;Git 命令的工作原理Git 如何工作Git merge vs. Git rebaseGit mergeGit rebaseGit rebaes 的黄金法则 云服务 : 不同云服务…

【C++】网络在线五子棋

项目介绍 本项目主要实现⼀个网页版的五⼦棋对战游戏&#xff0c;其主要⽀持以下核心功能&#xff1a; • 用户管理&#xff1a;实现用户注册&#xff0c;用户登录、获取用户信息、用户天梯分数记录、用户比赛场次记录等 • 匹配对战&#xff1a;实现两个玩家在网页端根据天梯分…

C++前缀和算法的应用:石头游戏 VIII 原理源码测试用例

本文涉及的基础知识点 C算法&#xff1a;前缀和、前缀乘积、前缀异或的原理、源码及测试用例 包括课程视频 题目 Alice 和 Bob 玩一个游戏&#xff0c;两人轮流操作&#xff0c; Alice 先手 。 总共有 n 个石子排成一行。轮到某个玩家的回合时&#xff0c;如果石子的数目 大…

ACM练习C++知识点笔记

1、字符和数字的转换 #include<iostream> using namespace std; int main(){int n 8 - 48;cout<<n<<endl;return 0; } 数字转字符串 #include <string> #include <sstream> #include <iostream> using namespace std; int main() {doubl…

基于Django开发的推荐系统与数据分析系统

基于Django开发的推荐系统与数据分析系统 一、简介 已开发的的推荐系统&#xff1a;图书管理系统、电影推荐系统、在线选修课程推荐系统、健身推荐系统、资讯推荐系统&#xff1b; 已开发的数据分析系统&#xff1a;大众点评店铺数据分析系统。 推荐系统的目的是信息过载所…

rockchip 3588 HDMI avmute

概述 HDMI (High-Definition Multimedia Interface) 是一种数字接口标准&#xff0c;用于传输高清视频和多通道音频信号。AVMUTE 是 HDMI 规范中的一个术语&#xff0c;表示"Audio-Video Mute"&#xff08;音视频静音&#xff09;。AVMUTE 通常与 HDMI 设备的音频和…