【Linux进程】进程的基本概念 {基本概念,Linux中的PCB—task_struct,查看进程,进程标识符—PID,创建子进程}

news2025/1/13 3:32:44

一、进程的基本概念

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

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

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

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

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

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

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

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

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

提示:在Linux系统中进程表是一个双向链表。


二、Linux中的PCB——task_struct

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

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

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

三、查看进程

3.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下的任务管理器
在这里插入图片描述


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

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

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

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

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

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

四、进程标识符——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


五、创建子进程

5.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

5.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给父进程返回子进程PID,给子进程返回0?

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

为什么fork能有两个返回值?

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

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

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

CPU运行队列

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

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

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

相关文章

vue3文件从创建 - 运行

新建文件 npm init vuelatest 2、报错处理 &#xff08;1&#xff09;TypeScript选项‘importsNotUsedAsValues‘和‘preserveValueImports‘弃用 在tsconfig.json中添加"ignoreDeprecations": "5.0" 参考&#xff1a;TypeScript选项‘importsNotUsedAs…

推荐10款Python图形界面(GUI)框架以及使用PyQt6和QtDesigner快速开发

9个国外最佳免费编程学习一站式网站 http://www.360doc.com/content/22/1024/17/63953942_1053111478.shtml QT文档 https://doc.qt.io/ QtDesigner pyqt6 https://blog.csdn.net/lyshark_lyshark/article/details/125847218 https://doc.qt.io/qtforpython-6/contents.htm…

途乐证券股票开户官网|CPO板块早盘大跌,两家近千亿巨头跌停

6月28日周三&#xff0c;A股集体低开&#xff0c;三大指数均一度跌超1%。 截至目前&#xff0c;沪指跌0.80%&#xff0c;深成指跌1.28%&#xff0c;创业板指跌0.91%。 盘面上&#xff0c;电力、煤炭开采加工等板块涨幅居前&#xff0c;泛AI、游戏、数据要素、光模块、传媒等方…

Stringbuffer的学习

1.构造方法 这里就看出String和Stringbuffer最主要的区别了 byte[]value不是final // 1.空参创建 这里 默认初始长度出现了&#xff0c;16public StringBuffer() {super(16); } // 2.定容创建&#xff0c;如果知道使用长度其实还好&#xff0c;因为他会扩容public StringBuf…

MySQL数据库——存储过程

MySQL数据库——存储过程 一、MySQL存储过程概念二、存储过程1.创建存储过程2.调用存储过程3.查看存储过程4.存储过程的参数5.删除存储过程 三、存储过程的控制语句1.条件语句if-then-else end if2.循环语句while end while 一、MySQL存储过程概念 存储过程是一组为了完成特定…

四、云尚办公-用户管理

云尚办公系统&#xff1a;用户管理 B站直达【为尚硅谷点赞】: https://www.bilibili.com/video/BV1Ya411S7aT 本博文以课程相关为主发布&#xff0c;并且融入了自己的一些看法以及对学习过程中遇见的问题给出相关的解决方法。一起学习一起进步&#xff01;&#xff01;&#x…

【2023,学点儿新Java-26】关键字介绍+示例代码:assert 断言(如何启用断言),以 验证一个数组的长度是否不为零 为例说明

前情回顾&#xff1a; 【2023&#xff0c;学点儿新Java-25】如何解决浮点计算存在误差&#xff1a;strictfp | 如何保护敏感信息&#xff0c;提高程序的可靠性和安全性&#xff1a;transient | 附&#xff1a;Java异常处理关键字介绍【2023&#xff0c;学点儿新Java-24】abstr…

yaffs文件系统测试

1.前面我制作好了文件系统烧到开发板运行后&#xff0c;出现了如下错误 2.网上查找后&#xff0c;有人说是因为 arm-linux-gnueabi-4.9xx版本太高&#xff0c;它的libc不支持arm9&#xff0c;于是我们换成arm-linux-gcc-4.3.2试试&#xff0c;arm-linux-gcc-4.3.2下载地址&…

【妙趣横生】用matplotlib绘制平均线,让你的数据可视化变得活灵活现!

大家好&#xff0c;今天我要跟你们分享一个超有趣的技巧——如何用matplotlib库绘制平均线&#xff0c;让你的数据可视化更加生动活泼&#xff01; 第一步&#xff1a;获取数据 我们首先需要将Excel表中的数据读取进来&#xff0c;这样才能绘制出精彩的图表。咱们可以使用pand…

在Fligma打开Android Material 3 Design组件规格颜色大小等

打开Material 3 Design Kit 链接&#xff1a;链接: Material 3 Design Kit 打开后点击右上方Open in Figma(提前注册Figma账号) 点击左侧方菜单栏中的Componenets选项查看M3组件规格

【JavaWeb】统一响应,并渲染在网页

一、需求描述 获取数据&#xff0c;返回统一响应结果&#xff0c;再在页面渲染展示。 二、知识点 1、在前端代码emp.html中&#xff0c;在页面加载时&#xff0c;通过vue的mounted方法发起异步请求来加载数据&#xff0c;在成功回调函数中判定响应回来的code的值&#xff0c…

个人总结 - IP代理池的思考

前言 今天话有点多&#xff0c;连续发了几篇博客&#xff0c;主要平常忙的话就没时间关注博客这块&#xff0c;今天兴致在&#xff0c;就勤快点哈哈 一般公司除非有钱&#xff0c;他可以购买ip服务器&#xff0c;或者大量高质量ip&#xff0c;但是有的时候&#xff0c;公司经…

高压放大器如何驱动压电陶瓷片

高压放大器是一种常用于驱动压电陶瓷片的电路&#xff0c;其基本原理是利用高压放大电路将低电压信号放大到足以驱动压电陶瓷片所需的高电压信号。在本文中&#xff0c;我们将介绍高压放大器如何驱动压电陶瓷片的具体方法和步骤。 图&#xff1a;压电陶瓷片 一、驱动压电陶瓷片…

三种方法教你:Allegro文件导入SIwave仿真

导入仿真工具进行信号完整性&#xff08;SI&#xff09;仿真是PCB设计中的关键步骤之一&#xff0c;但很多小白可能不太清楚该如何导入&#xff0c;下面将聊聊如何通过Allegro软件导入SIWave仿真&#xff0c;希望对小伙伴们有所帮助。 01 使用SIwave的直接导入功能 SIwave提供…

PADS 出gerber 20230628

PADS出gerber 20230628 TOP BOTTOM Solder MASK TOP 阻焊层 绿油层&#xff0c;可以用来露焊盘的 Paste Mask TOP 钢网层 Paste Mask Bottom Silkscreen TOP 丝印层 Solder MASK Bottom Silkcreen Bottom TOP层 L1 注意&#xff1a;电气层L1 L2 L3 L4不能勾选文本…

三维3D扫描仪工艺品摆件仿制翻模雕塑三维数字化3D打印-CASAIM

三维扫描技术在工艺品摆件仿制、翻模、3D打印、三维数字化方面发挥着重要作用。通过三维扫描技术&#xff0c;能完整还原工艺品的真实原貌&#xff0c;为复制经典艺术品提供了更精确和更环保的方法&#xff0c;最终精确保存细节、完美进行复制&#xff0c;并为以后的3D打印、三…

二叉树及其遍历方式!

二叉树 什么是二叉树&#xff1f; 树中每个节点最多只能有两个子节点&#xff0c;在 JavaScript 中一般都是通过 Object 来模拟二叉树。 常用操作 前序遍历中序遍历后序遍历 前序遍历 根左右。 口诀&#xff1a; 访问根节点对根节点的左子树进行前序遍历对根节点的右子…

使用python实现一个快速高斯模糊算法

在gimp的retinex里面使用了一个快速计算的高斯模糊&#xff0c;论文应该是Recursive Implementation of the gaussian filter&#xff0c;是使用一些多项式计算来近似计算高斯分布&#xff0c;这样能够大大减少计算。 将retinex源码给抽离出来&#xff0c;并使用python进行实现…

线性表的定义和基本操作(以顺序表为例)

名人说&#xff1a;一花独放不是春&#xff0c;百花齐放花满园。——《增广贤文》 作者&#xff1a;Code_流苏(CSDN)&#xff08;一个喜欢古诗词和编程的Coder&#x1f60a;&#xff09; 目录 〇、线性表是什么&#xff1f;1、定义2、特点3、基本操作 一、代码实现二、思路阐明…

统计图echarts和antd charts的那些事

以下记录的是&#xff0c;我在学习中的一些学习笔记&#xff0c;这篇笔记是自己学习的学习大杂烩&#xff0c;主要用于记录&#xff0c;方便查找&#xff0c;如需转载请著名原文作者和地址 一、echarts 相关 ##1、echarts 在react的使用 npm install echarts --save //初始化…