【Linux】进程(1)进程概念和进程状态

news2025/3/16 12:14:39

🌟🌟作者主页:ephemerals__

🌟🌟所属专栏:Linux

目录

前言

一、什么是进程

二、task_struct的内容

三、Linux下进程基本操作

四、父进程和子进程

1. 用fork函数创建子进程

五、进程状态

1. 三种重要状态

运行状态

阻塞状态

挂起状态

2. 内核链表的理解

3. Linux的进程状态

孤儿进程 

总结


前言

        在学习 Linux 操作系统的过程中,进程是一个至关重要的概念。无论你是想了解系统的基础操作,还是深入研究 Linux 内核,进程管理的理解都将为你打下坚实的基础。进程不仅是操作系统资源管理的核心,也是实现多任务处理的关键所在。通过学习进程的创建、调度、同步等机制,你可以更好地掌握操作系统的运行原理,进而优化系统性能和解决实际问题。本文将从基础知识入手,带领大家逐步深入探索 Linux 中进程的各个方面,帮助你在 Linux 学习的道路上迈出坚实的第一步。

一、什么是进程

         进程有多种描述方式,例如:程序的运行实例、正在执行的程序、操作系统进行资源调配的基本单位等。不过,以上说法都太理论化,我们用程序运行的实际情况来描述进程。

        一个程序在执行前,其二进制代码和数据(变量、常量、堆栈数据等)需要加载到内存。当加载完成之后,操作系统就会为这一块代码和数据创建一个对应的PCB(也叫做进程控制块,本质是一个存储进程相关信息的结构体),其中存在一个内存指针,指向代码和数据,便于访问。

        所以“进程”不仅仅包括了程序的运行实例,它也包括操作系统管理该进程的相关信息。简而言之,“进程”是指PCB与程序代码数据的集合操作系统根据PCB来跟踪进程的执行状态,方便对进程进行调度。

        而当有多个程序需要执行时,操作系统就会为每一个程序的代码和数据都创建一个对应的PCB(描述过程),再通过容器将所有的PCB串联起来(组织过程)。此时,操作系统对于进程的管理即为对容器的增删查改

需要注意:

在Linux下,PCB(进程控制块)是一个叫做task_struct的结构体;进程的所有属性都可以通过task_struct直接或间接地找到。

Linux下的task_struct之间通过双向链表进行连接。

二、task_struct的内容

 task_struct有如下成员,用于表示进程各种状态信息,以及访问程序的代码和数据:

  • 进程标识符(PID)--区别其他进程

  • 进程状态信息

  • 优先级

  • 程序计数器

  • 内存指针--指向代码和数据

  • 上下文数据

  • I/O状态信息

  • 记账信息

  • 其他信息

 之后的进程学习当中,我们将围绕以上成员数据,学习进程的相关概念及操作

三、Linux下进程基本操作


C语言函数获取当前进程标识符和父进程的标识符(PID):

getpid(); //返回当前进程标识符,返回值类型是pid_t
getppid(); //返回当前进程的父进程标识符

注意使用以上函数时,需要引头文件<unistd.h>


使用指令查看当前所有进程:

ps ajx
ls /proc

根据程序名查看某个进程信息:

ps ajx | head -1 && ps ajx | grep (可执行程序名)

根据标识符查看进程文件:

ll /proc/(标识符)

示例:

我们可以重点关注一下图中列举出的两个文件cwdexe

cwd指的是当前进程对应的可执行程序所在目录;

exe指的是当前进程对应的可执行程序位置。


C语言函数修改当前进程所在路径:

chdir("(路径)");

注意使用该函数要引头文件<unistd.h>


杀进程的两种方式:

1. ctrl + c

2. 命令行输入kill -9  (进程标识符)

四、父进程和子进程

        一个进程通过系统调用创建出的另一个进程称之为该进程的子进程,反之该进程称为其父进程。在Linux下,我们在命令行输入的命令都是Bash(命令行解释器)的子进程。

1. 用fork函数创建子进程

         fork是一个系统调用,存在于头文件<unistd.h>中,当执行fork函数之后,当前进程会创建一个子进程,后续的代码会被父进程和子进程分别执行一次

代码示例:

#include <stdio.h>
#include <unistd.h>

int main()
{
    fork();
    printf("hello world\n");
    return 0;
}

运行结果:

注意:fork函数创建的子进程没有自己的代码和数据,虽然操作系统为其创建了PCB,但是其内存指针指向的还是父进程的代码和数据。

 子进程在创建成功后,fork函数会给子进程返回0,给父进程返回子进程的PID。为什么会给父子进程不同的返回值呢?因为一个父进程可能会有多个子进程,给父进程返回子进程的PID,更方便父进程对子进程进行管理。而子进程如果想要知道父进程的PID,直接调用getppidh函数即可。另外,返回值不同可以配合分支语句让父子进程执行不同的代码。示例如下:

#include <stdio.h>
#include <unistd.h>

int main()
{
    pid_t id = fork();
    if(id == 0)//子进程
    {
        printf("我是子进程,我的pid是%d\n", getpid());
    }
    else//父进程
    {
        printf("我是父进程,我的pid是%d\n", getpid());
    }
    return 0;
}

运行结果:

        那么,为什么fork函数能够做到返回两个值呢?实际上fork函数在执行return语句之前,就已经创建好了子进程,此时就可以通过分支语句来区分给父进程和子进程的返回值。 

注意:虽然fork函数创建的子进程与父进程的代码是共享的,但如果父子任何一方要修改其中的数据,那么操作系统就会将数据进行拷贝,此时父子就各自维护自己的数据,本质上修改的是拷贝的数据,不会影响另一方。这种状况叫做写时拷贝

五、进程状态

        对于不同的操作系统,进程状态可能略有不同,但常见的大体上的进程状态有如下几种:创建、就绪、运行、阻塞、终止、挂起。我们介绍一下其中最重要的三点:运行状态、阻塞状态和挂起状态

1. 三种重要状态

运行状态

        首先要知道,一般情况下一个CPU维护一个进程调度队列,该队列中存放着一个个PCB,等待CPU对它们进行调度。而一个PCB在运行队列中排队时,就称该进程处于运行状态

阻塞状态

        当一个进程需要等待某种资源或设备(如鼠标、键盘等)就绪时,该进程就处于阻塞状态。阻塞状态的进程在代码层面的体现是:PCB从运行队列中移出,转而进入设备的等待队列当中

此时若设备准备就绪(如按下键盘),则操作系统会修改当前设备状态,然后检查等待队列,将等待队列中的PCB重新移动到运行队列当中,该进程重新恢复运行状态。

挂起状态

        当一个进程被暂停执行时,称该进程处于挂起状态。 那么它的具体体现是什么呢?

        当内存空间较为吃紧时,操作系统会将一些暂时不需要使用的内存数据(如阻塞状态的PCB控制的代码和数据唤出到磁盘中的swap交换分区。此时等待队列中的PCB不再维护该进程的代码和数据,这样的进程状态叫做阻塞挂起

        此时,若设备准备就绪,则操作系统就将swap交换分区中的代码和数据重新唤入到内存中,给PCB维护,然后恢复到运行状态。

        当内存空间严重不足时,操作系统会将运行状态的PCB控制的代码和数据也唤出到swap交换分区。此时称之为运行挂起


由这三种状态在代码层面的一部分具体体现,我们可以得出如下结论:进程状态的变化表现之一就是PCB在不同的数据结构之间移动,变化本质是操作系统对数据结构的增删查改

2. 内核链表的理解

        之前提到,在Linux下,操作系统会使用双向链表将PCB串联起来,方便进程管理。那么为什么PCB还会出现在CPU维护的调度队列当中呢?其实task_struct确实是同时出现在两种数据结构当中的,它基于一种特殊的结构来实现: 

task_struct当中,将用于构成双向链表的指针域封装成一个结构体list_head,它的指针指向的是其他task_struct的list_head。那么既然指向另一个指针域,如何能访问到task_struct的其他成员呢?这就需要用到结构体内存对齐的相关知识了:结构体的成员都是按照自身的对齐数进行存储的,第一个成员变量的地址就是结构体的首地址。通过求出list_head相对于结构体第一个成员的偏移量,就能间接访问结构体的其他成员。例如,如下表达式就可以表示next指针指向的list_head所在task_struct的首地址(其中links表示list_head的变量):

(struct task_struct*)(next - &((struct task_struct*)0->links))

将0强转为task_struct*类型,求出成员links的地址,即为links的偏移量,然后用links的地址减去该偏移量,得出task_struct的首地址,再强转为task_struct*类型,然后就可以访问其他成员了。

        而其他指针域也可以通过这种方式访问task_struct的其余成员,但可以用不同的链接方式,形成不同的数据结构,这样就实现了一个PCB同时存在于多种数据结构的壮举。

3. Linux的进程状态

        相比于之前提到的操作系统大体上的进程状态,Linux的进程状态就显得更加具体化。在Linux下,进程状态本质是task_struct内的长整型变量,它有以下几种进程状态表示:

static const char *const task_state_array[] = {
    "R (running)", /*0 */
    "S (sleeping)", /*1 */
    "D (disk sleep)", /*2 */
    "T (stopped)", /*4 */
    "t (tracing stop)", /*8 */
    "X (dead)", /*16 */
    "Z (zombie)", /*32 */
};

R:运行状态

S:休眠状态(可中断休眠)

D:深度睡眠状态(不可中断休眠)

进程处于深度睡眠状态时,不可被杀。

T:暂停状态--用户手动暂停进程(如Ctrl + z)

t:追踪状态--调试过程中执行到断点处,进程被暂停

x:死亡状态

z:僵尸状态--子进程在死亡之后,代码和数据可以释放,但其PCB不能直接释放,需要被父进程读取信息,读取信息之前称之为僵尸状态。

注意:如果父进程一直都不读取子进程的信息,那么僵尸状态就会一直存在,PCB也会一直存在,这就导致了内存泄漏。

孤儿进程 

        除了以上几种状态,进程还有一种特殊情况:孤儿进程。 当父进程先死亡,子进程就会被1号进程领养,成为新的父进程,此时该子进程就被称作孤儿进程。

注:1 号进程(init 或 systemd) 是 Linux 系统中的第一个用户态进程,负责初始化系统并管理其他进程。它由内核在系统启动时创建,PID固定为 1。现代 Linux 主要使用 systemd 作为 1 号进程,提供服务管理、日志收集和系统控制功能,而早期系统则使用 sysvinit 或 upstart。如果 1 号进程崩溃,系统通常会进入不可用状态,需要重启。

        那么为什么子进程会被1号进程领养呢?如果1号进程不领养它,则当子进程死亡后,没有父进程读取信息,就会造成内存泄漏

总结

        通过本篇文章,我们学习了Linux进程的基础知识,包括进程概念、task_struct 结构、进程状态以及父子进程关系,希望这篇文章能帮助你更清晰地理解Linux进程的运行机制。如果你觉得博主讲的还不错,就请留下一个小小的赞在走哦,感谢大家的支持❤❤❤

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

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

相关文章

关闭win11根据内容自动调整屏幕亮度

在win11笔记本上使用编程软件的时候&#xff0c;用的是深色背景&#xff0c;但是屏幕会慢慢变暗&#xff1b;等切换回明亮的桌面时&#xff0c;又会慢慢变亮&#xff0c;带来不适应的感觉。这个博客记录一下解决这个问题的办法 ps&#xff1a;有些人修改的是电源选项&#xff…

2021-05-23 C++百元百鸡

此是草稿&#xff0c;有值得优化的地方&#xff0c;如从公鸡先循环再母鸡再小鸡这样可以提高效率&#xff0c;且有输出后也可优化为公鸡母鸡小鸡初始化。 void 百元百鸡() {//缘由https://ask.csdn.net/questions/7434093?spm1005.2025.3001.5141int xj 1, mj 1, gj 1, y …

Android自动化测试工具

细解自动化测试工具 Airtest-CSDN博客 以下是几种常见的Android应用自动化测试工具&#xff1a; Appium&#xff1a;支持多种编程语言&#xff0c;如Java、Python、Ruby、JavaScript等。可以用于Web应用程序和原生应用程序的自动化测试&#xff0c;并支持iOS和Android平台。E…

【蓝桥杯】24省赛:数字串个数

思路 本质是组合数学问题&#xff1a; 9个数字组成10000位数字有9**10000可能 不包括3的可能8**10000 不包括7的可能8**10000 既不包括3也不包括77**10000 根据容斥原理&#xff1a;结果为 9 ∗ ∗ 10000 − 8 ∗ ∗ 10000 − 8 ∗ ∗ 10000 7 ∗ ∗ 10000 9**10000 - 8**10…

SpringBoot中使用kaptcha生成验证码

简介 kaptcha是谷歌开源的简单实用的验证码生成工具。通过设置参数&#xff0c;可以自定义验证码大小、颜色、显示的字符等等。 Maven引入依赖 <!-- https://mvnrepository.com/artifact/pro.fessional/kaptcha --><dependency><groupId>pro.fessional<…

蓝桥杯嵌入式赛道复习笔记1(led点亮)

前言 基础的文件创建&#xff0c;参赛资源代码的导入&#xff0c;我就不说了&#xff0c;直接说CubeMX的配置以及代码逻辑思路的书写&#xff0c;在此我也预祝大家人人拿国奖 理论讲解 原理图简介 1.由于存在PC8引脚到PC15引脚存在冲突&#xff0c;那么官方硬件给的解决方案…

六十天前端强化训练之第十七天React Hooks 入门:useState 深度解析

欢迎来到编程星辰海的博客讲解 看完可以给一个免费的三连吗&#xff0c;谢谢大佬&#xff01; 目录 一、知识讲解 1. Hooks 是什么&#xff1f; 2. useState 的作用 3. 基本语法解析 4. 工作原理 5. 参数详解 a) 初始值设置方式 b) 更新函数特性 6. 注意事项 7. 类组…

芯科科技推出的BG29超小型低功耗蓝牙®无线SoC,是蓝牙应用的理想之选

具有扩大的内存和超低功耗特性的超小型BG29是互联健康设备的理想之选 低功耗无线领域内的领导性创新厂商Silicon Labs&#xff08;亦称“芯科科技”&#xff0c;NASDAQ&#xff1a;SLAB&#xff09;今日宣布&#xff1a;推出全新的第二代无线开发平台产品BG29系列无线片上系统…

export、export default 和 module.exports 深度解析

文章目录 1. 模块系统概述1.1 模块系统对比1.2 模块加载流程 2. ES Modules2.1 export 使用2.2 export default 使用2.3 混合使用 3. CommonJS3.1 module.exports 使用3.2 exports 使用 4. 对比分析4.1 语法对比4.2 使用场景 5. 互操作性5.1 ES Modules 中使用 CommonJS5.2 Com…

qwen2.5-vl多机多卡分布式部署

记录一下工作中进行多机多卡部署qwen2.5-vl多模态大模型踩过的坑 第一个天坑就是官方提供的镜像qwenllm/qwenvl:2.5-cu121有问题&#xff0c;在titan显卡会抛出cuda error:no kernel image is availabe for execution on the device. 这是cuda内核与GPU不兼容的问题&#xff0c…

【红黑树】—— 我与C++的不解之缘(二十五)

前言 学习了avl树&#xff0c;现在来学习红黑树。 一、什么是红黑树 红黑树是一颗平衡二叉搜索树&#xff0c;它每一个节点增加了一个存储位表示节点的颜色&#xff0c;可以是红色或者黑色。 相比较于AVL树&#xff0c;红黑树也是一个自平衡二叉搜索树&#xff0c;但是它与AVL树…

驾驭 DeepSeek 科技之翼,翱翔现代学习新天际

在当今这个信息爆炸的时代&#xff0c;学习的方式和途径正在经历着前所未有的变革。人工智能技术的飞速发展&#xff0c;为我们的学习带来了全新的机遇和挑战。DeepSeek 作为一款强大的大语言模型&#xff0c;凭借其卓越的性能和丰富的功能&#xff0c;为现代学习注入了新的活力…

DeepSeek本地部署 (Windows+Ollama+Docker Desktop+ RAGFlow)

适用场景&#xff1a; 1、商城的小机器人自动根据实际情况回复 2、需要7*24小时运行在线回复&#xff0c;如&#xff1a;在线购物、在线咨询、在线招生等 3、无人值守环境 2025年1月&#xff0c;DeepSeek 正式发布 DeepSeek-R1 推理大模型&#xff0c;DeepSeek-R1 成本价格低…

SPI驱动(八) -- SPI_DAC设备驱动程序

文章目录 参考资料&#xff1a;一、编写设备树二、 编写驱动程序三、编写测试APP四、Makefile五、上机实验 参考资料&#xff1a; 参考资料&#xff1a; 内核头文件&#xff1a;include\linux\spi\spi.h内核文档&#xff1a;Documentation\spi\spidevDAC芯片手册&#xff1a;…

MySQL 衍生表(Derived Tables)

在SQL的查询语句select …. from …中&#xff0c;跟在from子句后面的通常是一张拥有定义的实体表&#xff0c;而有的时候我们会用子查询来扮演实体表的角色&#xff0c;这个在from子句中的子查询会返回一个结果集&#xff0c;这个结果集可以像普通的实体表一样查询、连接&…

HarmonyOS NEXT开发进阶(十二):build-profile.json5 文件解析

文章目录 一、前言二、Hvigor脚本文件三、任务与任务依赖图四、多模块管理4.1 静态配置模块 五、分模块编译六、配置多目标产物七、配置APP多目标构建产物八、定义 product 中包含的 target九、拓展阅读 一、前言 编译构建工具DevEco Hvigor&#xff08;以下简称Hvigor&#x…

深度学习笔记(37周)

目录 摘要 Abstracts 1. 介绍 2. 相关工作 3. 模型 3.1 时序段网络TSN 3.2 学习时序段网络 4. 训练结果 5. 结论 摘要 本周阅读的论文是《Temporal Segment Networks: Towards Good Practices for Deep Action Recognition》。作者主要想通过较少的训练样本&#xff…

ELK+Filebeat+Kafka+Zookeeper安装部署

1.安装zookeeper zookpeer下载地址:apache-zookeeper-3.7.1-bin.tar.gzhttps://link.csdn.net/?targethttps%3A%2F%2Fwww.apache.org%2Fdyn%2Fcloser.lua%2Fzookeeper%2Fzookeeper-3.7.1%2Fapache-zookeeper-3.7.1-bin.tar.gz%3Flogin%3Dfrom_csdn 1.1解压安装zookeeper软件…

【软考-架构】3.3、模式分解-事务并发-封锁协议

✨资料&文章更新✨ GitHub地址&#xff1a;https://github.com/tyronczt/system_architect 文章目录 模式分解&#xff08;难点&#xff09;无损分解&#x1f4af;考试真题并发控制封锁协议&#x1f4af;考试真题第一题第二题 模式分解&#xff08;难点&#xff09; 保持函…

审批工作流系统xFlow

WorkFlow-审批流程系统 该项目为完全开源免费项目 可用于学习或搭建初始化审批流程系统 希望有用的小伙伴记得点个免费的star gitee仓库地址 仿钉钉飞书工作审批流系统 介绍 前端技术栈: vue3 ts vite arcodesign eslint 后端技术栈:springbootspring mvc mybatis mavenmysq…