【Linux进程】进程的基本概念 {PCB结构体,进程表,Linux中的task_struct,查看进程,获取进程PID,使用fork创建子进程}

news2024/11/15 12:08:55

在这里插入图片描述

一、进程的基本概念

1.1 什么是进程?

进程是计算机中正在运行的程序的实例。它是操作系统进行资源分配和调度的基本单位。每个进程都有自己的内存空间、代码、数据和执行状态。进程可以独立运行,相互之间不会干扰。操作系统可以同时运行多个进程,通过分配时间片轮流执行它们,从而实现多任务处理。进程可以与其他进程进行通信和协作,共享资源和数据。

1.2 如何管理进程?

操作系统同样需要对进程进行管理。操作系统是怎样管理进程的呢?很简单,先把进程描述起来,再把进程组织起来!

  1. 描述进程:使用PCB结构体描述进程的各种属性

  2. 组织进程:使用数据结构将各进程的PCB结构体组织起来,形成进程表;再通过对进程表的增删查改实现对进程的管理。

要让存储在磁盘中的可执行文件变为进程,一是要将其代码和数据拷贝的内存中,二是要为其创建一个PCB结构体并初始化其属性,三是要将创建好的PCB加入到操作系统的进程队列。这样就形成了一个可被操作系统管理的进程了。

总结:进程=代码和数据+PCB结构体

1.3 PCB结构体、进程表是什么?

PCB:进程的属性信息被放在一个叫做进程控制块的数据结构中,可以理解为进程属性的集合。这类数据结构被称为PCB(process control block)。

提示:不同的操作系统有属于自己的不同的PCB。

进程表:进程表是操作系统中的数据结构,用于组织记录系统中所有进程的PCB。每个进程都有一个对应的进程表项(PCB),包含进程的标识符、状态、优先级、资源使用情况等信息。进程表可以用于管理和调度进程,操作系统可以根据进程表中的信息对进程进行分配资源、切换上下文等操作。

1.4 Linux中的PCB——task_struct

  • 在Linux中描述进程属性的PCB结构体叫做task_struct
  • task_struct是Linux内核的一种数据结构,它会被装载到RAM(内存)里并且包含着进程的属性信息。
  • Linux内核使用双向链表组织和维护所有进程的PCB,用于管理和调度进程。

task_ struct中记录的进程属性有:

  • 标示符(PID): 描述本进程的唯一标示符,用来区别其他进程。
  • 状态: 任务状态,退出代码,退出信号等。
  • 优先级: 相对于其他进程的优先级。
  • 程序计数器: 程序中即将被执行的下一条指令的地址。
  • 内存指针: 包括程序代码和进程相关数据的指针,还有和其他进程共享的内存块的指针
  • 上下文数据: 进程执行时处理器的寄存器中的数据[休学例子,要加图CPU,寄存器]。
  • I/O状态信息: 包括显示的I/O请求,分配给进程的I/O设备和被进程使用的文件列表。
  • 记账信息: 可能包括处理器时间总和,使用的时钟数总和,时间限制,记账号等。
  • 其他信息

二、查看进程

2.1 使用ps或top工具获取进程信息

ps命令:查看当前终端的所有进程 加axj选项 查看系统所有进程

在这里插入图片描述

以下是一个持续监控指定进程的脚本:

[zty@192 ~]$ while :; do ps axj | head -1 && ps axj | grep myproc | grep -v grep; echo "###########################"; sleep 1; done

这段代码是一个无限循环的脚本。它的作用是每隔1秒钟打印出当前正在运行的进程中包含"myproc"关键字的进程信息

具体解释如下:

  1. while :; do:表示开始一个无限循环,:是一个永远返回true的命令,所以这个循环会一直执行下去。

  2. ps axj | head -1:通过ps axj命令获取当前所有进程的信息,并通过head -1命令只取第一行,即进程信息的标题行。这一行会被打印出来。

  3. ps axj | grep myproc | grep -v grep:通过ps axj命令获取当前所有进程的信息,并通过grep myproc过滤出包含"myproc"关键字的进程信息,然后通过grep -v grep去除掉包含"grep"关键字的行。这样就只剩下包含"myproc"关键字的进程信息了,它们会被打印出来。

  4. echo "###########################":打印一行分隔符,用于区分不同时间点的进程信息。

  5. sleep 1:暂停1秒钟,等待下一次循环。

top命令:查看系统所有进程,类似于windows下的任务管理器
在这里插入图片描述


2.2 通过proc目录查看进程信息

进程的信息可以通过/proc 系统目录查看

系统中进程的属性数据都保存在/proc下以其PID为名的目录下:在这里插入图片描述

/proc目录下的文件是动态的,每创建进程就会新增一个PID目录,进程结束该PID目录同时也会被删除。

PID目录如下:
在这里插入图片描述

  • cwd:当前进程的工作目录,可执行程序的所在目录,程序中读写文件的默认路径。
  • exe:可执行程序的路径

三、获取进程PID

3.1 什么是进程PID?

  1. 进程PID(Process ID)是操作系统为每个正在运行的进程分配的唯一标识符。 PID是一个整数值,用于在操作系统中唯一标识一个进程。每当一个新的进程被创建时,操作系统会为其分配一个唯一的PID。
  2. PID在操作系统中起到了重要的作用,它可以用来识别和管理进程。通过PID,操作系统可以追踪进程的状态、资源使用情况以及与其他进程的关系。PID还可以用于进程间通信、进程调度和资源分配等操作。
  3. 在Linux中,进程PID的数据类型是pid_t,它定义在<sys/types.h>头文件中。pid_t类型是一个有符号整数类型,通常是int类型。
  4. 需要注意的是,虽然PID是有符号整数类型,但在实际使用中,PID的值始终是非负整数,不会出现负数的情况。

在这里插入图片描述

3.2 如何获取进程PID?

getpid:获取调用进程的PID

  • 头文件<unistd.h>和<sys/types.h>是由Linux操作系统提供的,和语言无关。
  • pid_t:由操作系统提供的数据类型,实际上是一个无符号整数。

kill -9 pid:向指定进程发送9号指令,终止指定进程。

getppid:获取父进程的PID

  • 无论是直接调用系统命令还是通过./运行自己编写的程序,其父进程都是bash

  • bash会通过创建子进程来执行命令

  • 每一次登录,都会创建一个独属于当前终端的bash。

测试代码:

void Test1(){                                                                                                                    
  while(1)    
  {    
    pid_t pid = getpid();    
    pid_t ppid = getppid();    
    cout << "hello world!";    
    cout << " pid:" << pid;    
    cout << " ppid:" << ppid << endl;    
    sleep(1);    
  }    
}  

运行结果:
在这里插入图片描述

在这里插入图片描述

每次运行程序,进程的pid都不同,但其父进程ppid都是9031——bash


四、创建子进程

4.1 初识fork

在这里插入图片描述

在这里插入图片描述

测试代码:

void Test2(){                                                                                
  cout  << "I'm parent process:" << getpid() << endl;    
  pid_t ret = fork();    
  //下面的代码由父进程和子进程执行    
  cout << "ret:" << ret << " ";    
  cout << "pid:" << getpid() << " ";       
  cout << "ppid:" << getppid() << endl;    
}  

运行结果:

在这里插入图片描述

  1. fork之后的代码由两个进程执行,一个是父进程,一个是子进程
  2. fork创建子进程成功返回给父进程子进程的PID,返回给子进程0
  3. fork创建进程失败返回-1

4.2 fork的基本用法

  • fork之后的代码是父子进程共享的
  • 可通过判断fork的返回值使用分支语句对父子进程进行分流,即让父子进程执行不同的代码。

测试代码:

int Test1(){
  pid_t id = fork();
  if(id < 0)
  {
    //创建失败
    cerr << "fork" << endl;
    return 1;
  }
  else if(id == 0)
  {
    //子进程执行流
    while(1)
    {
      cout << "I'm child process, ";
      cout << "pid:" << getpid() << " ";
      cout << "ppid:" << getppid() << endl;                                                    
      sleep(1);
    }
    return 0;
  }
  else{
    //父进程执行流
   while(1)
    {
      cout << "I'm parent process, ";
      cout << "pid:" << getpid() << " ";                                                       
      cout << "ppid:" << getppid() << endl;
      sleep(1);
    }
    return 0;
  }
}

运行结果:

在这里插入图片描述
为什么fork能有两个返回值?

  1. fork是一个系统调用接口(C函数),由Linux内核实现
  2. 在fork内部,return之前子进程就已经被创建出来了。父子进程会执行各自的return语句,因此会有两个返回值。
  3. 两个返回值返回到两个不同的进程;一个进程中只有一个确切的返回值。

为什么fork给父进程返回子进程PID,给子进程返回0?

  1. 因为一个父进程可以有多个子进程,所以需要子进程PID对子进程进行标识。而每个子进程都只有一个父进程,所以不需要父进程PID,返回0即可。
  2. 代码中需要区分父子进程以进行分流操作,因此不能返回父进程PID。

创建出子进程后,父子进程哪一个先运行呢?

  1. 创建子进程的本质是以父进程为模版创建一个子进程的test_struct,并加入到进程表,交由操作系统进行进程管理。
  2. 操作系统会创建一个CPU运行队列,调度器会根据进程的优先级,状态等信息,将进程的test_struct加入到运行队列。CPU会依次执行队列中的进程
  3. 无法确定父子进程的运行顺序,因为进程运行的先后顺序有多种影响因素,最终由操作系统的调度器决定。

CPU运行队列

  • CPU运行队列是操作系统中的一个数据结构,用于存放等待在CPU上执行的进程。当操作系统决定将CPU分配给某个进程时,该进程会被放入CPU运行队列中等待执行。 CPU运行队列可以根据不同的调度算法来确定进程的执行顺序,例如先来先服务、短作业优先、时间片轮转等。
  • 进程表和CPU运行队列是相关但不完全相同的概念。简而言之,进程表是记录系统中所有进程的信息的数据结构,而CPU运行队列是存放等待在CPU上执行的进程的数据结构。进程表可以用于管理和调度进程,而CPU运行队列则是用于确定进程的执行顺序

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

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

相关文章

vue表格实现一个简单的合并单元格功能

用的是vue2ant-design-vue 但是vue3或者element-ui也是同理 先上效果 需要后端的数据将相同id的放在一起 否则也会有问题 例如&#xff1a; this.list [{id: 1,name: 舟山接收站,...}{id: 2,name: 舟山接收站碳中和LNG,...},{id: 2,name: 舟山接收站碳中和LNG,...} ]// th…

Redis7【⑤ Redis 发布 订阅】

Redis发布和订阅 本章了解即可&#xff0c;命令可以不用敲。 Redis 发布和订阅&#xff08;Publish/Subscribe&#xff0c;简称 Pub/Sub&#xff09;是一种消息传递模式&#xff0c;用于在 Redis 中实现消息的发布和订阅。 在 Redis 中&#xff0c;发布者&#xff08;Publi…

maven打包所有依赖,对外提供sdk.jar

maven打包所有依赖 <properties><project.build.sourceEncoding>UTF-8</project.build.sourceEncoding><maven.compile.source>1.8</maven.compile.source><maven.compile.target>1.8</maven.compile.target></properties><…

Swin Transformer训练报错问题

1. 训练遇到报错问题 &#xff08;1&#xff09;mportError: cannot import name _pil_interp from timm.data.transforms 原因&#xff1a; timm.data.transforms里面没有_pil_interp&#xff0c;只有str_to_pil_interp、_str_to_pil_interpolation、_pil_interpolation_to_s…

rancher 节点重启无感发布

这里设置 时间 为120s &#xff0c;保证 新节点起来后&#xff0c;和 老节点并行2分钟后再剔除&#xff0c;老节点

el-select修改样式

目录 准备 修改placeholder颜色 修改右侧箭头 修改圆角边框 准备 <el-select v-model"goodsId" clearable placeholder"请选择" :popper-append-to-body"false"><el-option v-for"item in kindList" :key"item.value…

浙江宇视科技 网络视频录像机 ISC LogReport.php 远程命令执行漏洞

免责声明 文章仅供参考&#xff0c;任何个人和组织使用网络应当遵守宪法法律&#xff0c;遵守公共秩序&#xff0c;尊重社会公德&#xff0c;不得危害网络安全&#xff0c;不得利用网络从事危害国家安全、荣誉和利益&#xff0c;未经授权请勿利用文章中的技术资料对任何计算机…

Vue Router activated deactivated 路由守卫

6.12.activated deactivated activated和deactivated是路由组件所独有的两个钩子&#xff0c;用于捕获路由组件的激活状态具体使用 activated路由组件被激活时触发deactivated路由组件失活时触发 src/pages/News.vue <template><ul><li :style"{opacity}…

1.2-程序设计语言与流程图基础

一、学习目标 了解计算机程序与程序设计语言。认识算法和流程图。理解计算机程序、程序设计语言、算法与流程图之间的关系。 1、计算机程序 计算机程序是人们使用指定的程序设计语言&#xff0c;根据需要事先编写的一系列控制计算机工作的命令。2、程序设计语言 程序设计语…

如何避免死锁:方法一

需要先看前文&#xff1a;死锁的产生_御坂美琴1的博客-CSDN博客 对两个资源使用一把锁。即小朋友玩敲鼓的时候会同时拿走鼓和鼓槌。 如图&#xff1a; 可以看到“线程1执行了&#xff0c;但是线程2没有执行&#xff0c;还在被阻塞着。为什么线程1运行完毕&#xff0c;线程2还没…

解决sourcetree中推送不显示分支 - 软件篇

问题&#xff1a; 如上图所示&#xff0c;在我们使用sourcetree推送我们的代码时 推送的弹窗里不回显示我们的分支&#xff1b; 众所周知&#xff1a;在我们没有新创建分支的情况下&#xff0c;他会默认有一个master分支&#xff0c;这也是我们的主分支 在网上搜这个问题&…

61 KVM Skylark虚拟机混部-使用及最佳实践

文章目录 61 KVM Skylark虚拟机混部-使用及最佳实践61.1 使用Skylark61.1.1 启动服务61.1.2 创建虚拟机61.1.3 虚拟机运行 61.2 最佳实践61.2.1 虚拟机业务推荐61.2.2 虚拟机绑核配置 61 KVM Skylark虚拟机混部-使用及最佳实践 61.1 使用Skylark 61.1.1 启动服务 初次启动&a…

Spring MVC的工作原理

1.Spring MVC的工作原理 1.浏览器发送请求&#xff0c;DispatcherServlet拦截请求 2.DispatcherServlet将需要先调用HandlerMapping通过uri找到能够处理请求的Handler&#xff0c;然后将请求涉及到的Handler封装 3.DispatcherServlet调用HandlerAdapter适配执行Handler 4.H…

LeetCode 打卡day45-- 单词拆分和多重背包问题

一个人的朝圣 — LeetCode打卡第45天 知识总结 Leetcode 139. 单词拆分题目说明代码说明 知识总结 今天写了一道题目, 但是还挺难的, 而且是面试高频题目 还过了一遍多重背包问题. 多重背包与01背包的区别在于多重背包限制了物品的个数, 某些物品的个数可能不为1, 可以使用两…

Java缓存简介

一、缓存 1、什么是缓存&#xff1f; 缓存是硬件&#xff0c;是CPU中的组件&#xff0c;CPU存取数据的速度非常的快&#xff0c;一秒钟能够存取、处理十亿条指令和数据&#xff08;术语&#xff1a;CPU主频1G&#xff09;&#xff0c;而内存就慢很多&#xff0c;快的内存能够达…

数据库实验-图书销售管理系统数据库安全管理

一、实验二&#xff1a;图书销售管理系统数据库安全管理 三、实验目的 了解该DBMS系统对数据库管理的内容与方法&#xff0c;特别是理解数据库安全机制和作用&#xff0c;以及PostgreSQL数据库角色管理、用户管理、权限管理的基本方法&#xff0c;培养数据库管理能力。在图书…

CV什么时候能迎来ChatGPT时刻?

卷友们好&#xff0c;我是rumor。 最近看了几篇CV的工作&#xff0c;肉眼就感受到了CVer们对于大一统模型的“焦虑”。 这份焦虑让他们开始尝试统一一切&#xff0c;比如&#xff1a; 统一复杂的自动驾驶任务的优化目标[1]&#xff0c;来自今年CVPR最佳论文。统一典型的CV任务&…

3s定时循环启动弹窗,子窗体3s后自动关闭功能实现

只贴关键代码&#xff1a; 1、父窗体关键代码 private void doAt1AM(object state){//要执行的任务&#xff0c;直接button调用即可//执行功能...的任务spring_Form spr_Form new spring_Form();spr_Form.Txt "陈先生" ";到1号窗口取药";spr_Form.Show…

WPF中的Behavior及Behavior在MVVM模式下的应用

WPF中的Behavior及Behavior在MVVM模式下的应用 在WPF中&#xff0c;Behaviors&#xff08;行为&#xff09;是一种可重用的组件&#xff0c;可以附加到任何UI元素上&#xff0c;以添加特定的交互行为或功能。Behaviors可以通过附加属性或附加行为的方式来实现。 Behavior并不…

【3Ds Max】使用捕捉工具、轴约束工具实现点的精准对齐

目录 问题 解决步骤 一、在XYZ轴上都对齐 二、只在一或两个轴上对齐 问题 举个例子&#xff0c;比如我想让如下的两个立方体的顶点对齐&#xff08;在同一位置&#xff08;XYZ轴都对齐&#xff09;&#xff0c;或仅在一个或两个轴上对齐&#xff09;&#xff1a; 主要使…