Linux进程概念僵尸进程孤儿进程

文章目录

    • 一、什么是进程
    • 二、进程的状态
    • 三、Linux是如何做的?
      • 3.1 R状态
      • 3.2 S状态
      • 3.3 D状态
      • 3.4 T状态
      • 3.5 t状态
      • 3.6 X状态
      • 3.7 Z状态
    • 四、僵尸进程
      • 4.1 僵尸进程危害
    • 五、孤儿进程

一、什么是进程

  • 对于进程理解来说,在Windows上是也可以观察到的,右键状态栏的任务管理器就可以看到

在这里插入图片描述

  • 运行一个进程是需要占用CPU和内存的
  • 在我们平时运行一个进程的时候(Windows)直接双击后,系统将磁盘里的exe文件加载到内存当中,这个时候就当运行起来的时候就开启了一个进程,这里要注意,加载到内存中中是不算进程的,只有当运行起来后才算是进程 -->进程 = PCB + 自己的代码和数据
  • 对应的每一个进程都有一个PCB结构体,操作系统需要通过此结构体找对应的二进制
  • 当有多个进程被运行起来的时候,就会产生多个PCB结构体,这个时候需要组织(可以是链表或者其他更高效的数据结构)起来进行管理,这就产生一个重要的概念------->先描述,再组织

在这里插入图片描述

  • 每个进程都对应的每个PCB结构体,当运行的时候需要进行排队,CPU就可以通过PCB找到原代码,再把代码交给CPU运行

在这里插入图片描述

二、进程的状态

  • 每个进程都有自己的一个状态,告诉操作系统正在干什么,将来要干什么?也就是说,进程的多种状态,本质都是为了满足未来的某种使用场景。

  • 就比如Windows上的任务管理器也有显示进程的状态

在这里插入图片描述


  • 在操作系统上有三种状态
    • 运行状态
    • 挂起状态
    • 阻塞状态

  • 运行状态:从计算机的硬件出发,我们所写的代码在硬盘中,要让程序运行起来就要加载到内存当中, 每一个程序(进程)都会有一个属于自己的PCB,通过PCB来进行排队,等待CPU的调度,为了方便调度管理,操作系统会维护一个运行队列,所有就绪状态的进程的PCB会被加入到这个队列当中, CPU在调度执行时就会通过这个运行队列拿到进程的PCB,进而调度执行该进程,在排队的时候就是运行状态。

  • 挂起状态:内存满负荷时,又要增加新的进程显然是不行的。所以操作系统会观察内存中的哪些进程没有被放在任何一个队列里面(在内存里面啥也不干),找到以后就把此进程的代码和数据短期内置换到磁盘上,仅保留此进程的PCB。腾出的这一块空间供新的进程使用。针对于这种情况,操作系统会将阻塞进程的代码和数据置换到外设,此时该进程的状态就被称为挂起状态;

    • 其中阻塞进程的代码和数据一般会存放在磁盘的swap分区,当进程被操作系统调度时,被置换到外设的代码和数据又会重新加载到内存;

    • 一般情况下,swap分区的大小不会太大,大概等于内存的大小,过大的swap分区会导致操作系统过于依赖swap分区,导致效率变低;


  • 阻塞状态:在CPU执行一个进程的时候,可能会需要访问系统的某些资源,就比如在C语言中写的scanf(),在使用这个函数的时候,需要调用键盘,等待键盘输入数据,当进程需要键盘资源的时候,会将进程的PCB加入到硬件设备结构描述的等待队列当中,并且把PCB设置为阻塞状态,当PCB在这个等待队列中等待数据资源时,这个状态就叫做阻塞状态;

三、Linux是如何做的?

  • 在Linux内核中是这样定义的:
/*
* The task state array is a strange "bitmap" of
* reasons to sleep. Thus "running" is zero, and
* you can test for combinations of others with
* simple bit tests.
*/
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 (Running):该进程正在运行中。
    • S (Sleep):该进程目前正在睡眠状态,但可以被唤醒。
    • D :不可被唤醒的睡眠状态,通常这个进程可能在等待I/O的情况。
    • T :停止状态,发送一个暂停信号给进程,进程就暂停了。
    • t :追踪停止状态,通常在断点调试时,进程处于此状态。
    • X :死亡状态,这个状态是用来告诉操作系统的,所以我们观察不到此状态。
    • Z (Zombie):僵尸状态,进程已经死亡,但是却无法被删除至内存外。

  • 下面我们挨个介绍一下,以下用Linux来查看:

在这里插入图片描述

  • 这里我们还需要了解一个概念,进程是有依赖的,就比如我运行这个C语言程序,其父进程是bash,pid是本进程的id,ppid是父进程的id

在这里插入图片描述

3.1 R状态

  • 下面我们编写这样一段代码,是一个死循环,这样可以观察到进程
#include<stdio.h>
#include<unistd.h>
int main()
{while(1){}return 0;
}
  • 普通查看进程可以这样查看
ps -aux | head -1 &&ps -aux | grep a.out

在这里插入图片描述

  • 我们还可以用shell命令来一直刷新观察
while :; do ps ajx | head -1 && ps ajx | grep a.out | grep -v grep; sleep 1; done

在这里插入图片描述

  • 这里看到是R+是代表在前台运行,在运行的时候可以使用ctrl+c进行中断,中断后进程也就退出了

  • 我们可以让此程序到后台运行

    • 在命令后面加“&”符号

    • 这里的pid进程id是唯一的,标识此进程,ppid代表父进程

在这里插入图片描述

  • 在运行起来后看到那个+号就没有了,说明是在后台运行中

在这里插入图片描述

  • 想要杀死这个程序就需要使用kill命令来进行杀死
kill -9 27165
  • 这里使用kill -9命令比较麻烦需要找到进程的id,我们还有一个命令
killall -9 a.out
  • 这样就可以不用输入进程id输入此进程的名字即可

  • 我们再补充一个概念,每一个进程运行起来的时候都有一个文件被创建

  • 在根目录下的proc

在这里插入图片描述


3.2 S状态

  • 这个状态表示此进程正在睡眠,但是可以被唤醒

  • 下面可以进行演示一下:

#include<stdio.h>
#include<unistd.h>
int main()
{while(1){printf("hello world\n");}return 0;
}
  • 在上面的那段代码中添加了一个printf打印函数,同样是运行,但是这里为什么是S状态呢?

在这里插入图片描述

  • 这是因为CPU运行的速度很快,一直需要请求显示器响应,CPU多数的时候是在等待,所以就显示S

  • 这里我使用了实体机器的ubuntu验证了一下,是对的,一遍在疯狂的打印helloword,由于打印的速度也很快,CPU的速度也很快,所以有再看进程有的时候就会显示R状态,有的时候显示S状态~~

在这里插入图片描述


3.3 D状态

  • 此状态是深度休眠状态,不可被唤醒,是为磁盘准备的

  • 当进程需要大量的数据写入磁盘的时候,等待磁盘写入的进程状态就是休眠,上面也将过了,

  • 在内存严重不足的情况下,操作系统没办法时会通过杀死进程的方式来节省资源;如果在等待的过程中进程被操作系统杀掉,并且磁盘写入数据失败,那么就会导致数据无法再加载(数据丢失);为了避免这种情况,就可以把等待数据写入的进程状态设为D状态;

  • D状态无法被杀掉(OS也不行),只能等待执行完毕后状态转换;


这个D状态不好验证,如果看到此状态说明计算机就要快挂了~~


3.4 T状态

  • 此状态是停止状态,可以通过发送 SIGSTOP 信号给进程来停止(T)进程。这个被暂停的进程可以通过发送 SIGCONT 信号让进程继续运行

在这里插入图片描述

在这里插入图片描述

  • 可以使用kill -l命令查看一下发送信号的类型

在这里插入图片描述


3.5 t状态

  • 这个小写的t状态也是停止状态

  • 主要出现在程序Debug时,在Debug的时候,遇到断点,进程就暂停,此时就是t状态

  • D状态、T状态、t状态其实都是阻塞状态,阻塞可以等待硬件资源也可以等待软件资源


3.6 X状态

  • X状态(死亡状态),就是我们常说的终止状态,它是一个瞬时状态,不会在任务列表里看到这个状态

3.7 Z状态

  • 最后就是Z(zombie)僵尸状态,僵尸状态较为复杂,Linux系统中的僵尸进程状态也是一种特殊的进程状态,通常是指一个子进程已经结束运行,但其父进程还未对其进行善后处理。如果不及时清理僵尸进程,会导致系统资源泄漏,影响系统性能甚至造成系统崩溃;

四、僵尸进程

  • 当有一个进程要退出的时候,它是直接原地消失、释放空间的吗?

  • 当进程退出的时候,它是不会立即释放空间的,它的PCB会保存一段时间让父进程或者操作系统读取,让父进程或操作系统知道这个进程即将退出了,然后父进程或者操作系统释放掉进程占用的资源和空间。一般情况下,清理进程资源空间的操作都是父进程。

  • 那么僵尸进程,指的是什么呢?就是进程退出时,依然会在内存里面待一段时间,如果父进程没有能力将此进程完整地释放掉,造成这个进程一直在内存里面,此时这个进程就是僵尸进程


  • 下面我们创建一个子进程来看一看
#include<stdio.h>
#include <sys/types.h>
#include <unistd.h>
int main()
{pid_t id = fork();       while(1){if(id == 0){printf("我是子进程,pid:%d,ppid:%d\n",getpid(),getppid());sleep(1);}else if(id > 0){printf("我是父进程,pid:%d,ppid:%d\n",getpid(),getppid());sleep(2);}}                                                                                  return 0;
}
  • 使用man 的2号手册可以看到fork的用法,fork有两个返回值
  • 父子进程代码共享,数据各自开辟空间,私有一份(采用写时拷贝)

在这里插入图片描述

  • return val这里可以看到返回值,如果成功返回0,失败返回-1,

  • 在程序上面的那个程序的时候,杀死子进程后,子进程的状态就变成了僵尸进程

在这里插入图片描述

  • 僵尸进程通常是无法再进行管理的,所以我们不能直接杀kill掉它,而是交给操作系统来处理这个进程。

4.1 僵尸进程危害

  • 进程的退出状态必须被维持下去,因为他要告诉关心它的进程(父进程),你交给我的任务,我办的怎么样了。可父进程如果一直不读取,那子进程就一直处于Z状态?是的!
  • 维护退出状态本身就是要用数据维护,也属于进程基本信息,所以保存在task_struct(PCB)中,换句话说,Z状态一直不退出,PCB一直都要维护?是的!
  • 那一个父进程创建了很多子进程,就是不回收,是不是就会造成内存资源的浪费?是的!因为数据结构对象本身就要占用内存,想想C中定义一个结构体变量(对象),是要在内存的某个位置进行开辟空间!

五、孤儿进程

  • 孤儿进程,顾名思义,没有父亲的进程就是孤儿进程,也就是,父进程创建了一个子进程,而父进程先退出了,就留下了子进程,这个时候这个进程就叫做孤儿进程

  • 我们再用上面的代码可以演示一下,运行起来后,杀死父进程,就留下一个子进程

在这里插入图片描述

  • 父进程是2200,当杀死父进程后,子进程的ppid就变成了1,就被操作系统接管了

本文介绍了Linux的进程概念,补充了一些概念,最后讲解了僵尸进程和孤儿进程是什么原因造成的

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

ManualResetEvent 在线程中的使用C#

ManualResetEvent 用于表示线程同步事件&#xff0c;可以使得线程等待信号发射之后才继续执行下一步&#xff0c;否则一直处于等待状态中。 ManualResetEvent 的常用方法 构造函数ManualResetEvent(bool); ManualResetEvent manualResetEvent new ManualResetEvent(false…

16. 获取社区详情

文章目录 一、建表、插入测试数据、并生成对应的model二、添加路由&#xff0c;以及controller、logic、dao三层分别实现对应代码三、编译运行 登录之后&#xff0c;我们可以发表帖子&#xff0c;但是发表帖子之前&#xff0c;需要先选择一个频道&#xff0c;可以理解是社区分类…

什么是高级编程语言?——跟老吕学Python编程

什么是高级编程语言&#xff1f;——跟老吕学Python编程 高级编程语言简介高级编程语言发展历程高级编程语言特点高级编程语言分类命令式语言函数式语言逻辑式语言面向对象语言 常见的高级编程语言及其特点和应用领域高级编程语言性能分析高级编程语言的工作方式 高级编程语言简…

MUMU模拟器12连logcat的方法

大家好&#xff0c;我是阿赵。   在开发手机游戏的时候&#xff0c;在真机上会出现各种问题&#xff0c;在查询问题的时候&#xff0c;安卓手机需要用adb连接来连接手机看logcat输出分析问题。但由于连接手机比较麻烦&#xff0c;所以我都习惯在电脑用安卓模拟器来测试。   …

R语言基础的代码语法解译笔记

1、双冒号&#xff0c;即&#xff1a;“::” 要使用某个包里的函数&#xff0c;通常做法是先加载&#xff08;library&#xff09;包&#xff0c;再调用函数。最新加载的包的namespace会成为最新的enviroment&#xff0c;某些情况下可能影响函数的结果。而package name::funct…

角蜥优化算法 (Horned Lizard Optimization Algorithm ,HLOA)求解无人机路径优化

一、无人机路径规划模型介绍 无人机三维路径规划是指在三维空间中为无人机规划一条合理的飞行路径,使其能够安全、高效地完成任务。路径规划是无人机自主飞行的关键技术之一,它可以通过算法和模型来确定无人机的航迹,以避开障碍物、优化飞行时间和节省能量消耗。 二、算法介…

Appium系列(1)安装启动Appium

Appium环境准备 Mac电脑jdk环境AndroidSDK环境node>8.1.0&#xff08;最好用最新版本&#xff09; 安装命令 npm i -g appium安装不成功请检查node 版本是否正确 安装成功命令行输入appium回车查看 安装驱动程序 1、先检查当前驱动情况 通过 appium driver list 进行…

【go语言开发】redis简单使用

本文主要介绍redis安装和使用。首先安装redis依赖库&#xff0c;这里是v8版本&#xff1b;然后连接redis&#xff0c;完成基本配置&#xff1b;最后测试封装的工具类 文章目录 安装redis依赖库连接redis和配置工具类封装代码测试 欢迎大家访问个人博客网址&#xff1a;https://…

系统学习c++类和对象——深度理解默认成员函数

前言&#xff1a;类和对象是面向对象语言的重要概念。 c身为一门既面向过程&#xff0c;又面向对象的语言。 想要学习c&#xff0c; 首先同样要先了解类和对象。 本节就类和对象的几种构造函数相关内容进行深入的讲解。 目录 类和对象的基本概念 封装 类域和类体 访问限定符…

Everything:文件查找工具,一搜即得

名人说&#xff1a;东边日出西边雨&#xff0c;道是无晴却有晴。——刘禹锡 创作者&#xff1a;Code_流苏(CSDN)&#xff08;一个喜欢古诗词和编程的Coder&#x1f60a;&#xff09; 目录 一、软件介绍①Everything②核心功能③原理 二、下载安装①下载②安装 三、使用方法①文…

吴恩达机器学习笔记十六 如何debug一个学习算法 模型评估 模型选择和训练 交叉验证测试集

如果算法预测出的结果不太好&#xff0c;可以考虑以下几个方面&#xff1a; 获得更多的训练样本 采用更少的特征 尝试获取更多的特征 增加多项式特征 增大或减小 λ 模型评估(evaluate model) 例如房价预测&#xff0c;用五个数据训练出的模型能很好的拟合这几个数据&am…

Linux调试器--gdb的介绍以及使用

文章目录 1.前言 ✒️2.介绍gdb✒️3.Debug模式和Release模式的区别✒️4.如何使用gdb✒️1️⃣.在debug模式下编译2️⃣.进入调试3️⃣ .调试命令集合⭐️⭐️ 1.前言 ✒️ &#x1f557;在我们之前的学习中已经学会了使用vim编译器编写c/c代码&#xff0c;但是对于一个程序员…

OpenCV 图像的几何变换

一、图像缩放 1.API cv2.resize(src, dsize, fx0,fy0,interpolation cv2.INTER_LINEAR) 参数&#xff1a; ①src &#xff1a;输入图像 ②dsize&#xff1a;绝对尺寸 ③fx&#xff0c;fy&#xff1a;相对尺寸 ④interpolation&#xff1a;插值方法 2.代码演示 import cv2 …

Python——读写属性

采用读写属性的目的就是把录入的数据控制在合理区间。 如&#xff1a;学生的年龄&#xff08;age&#xff09;&#xff0c;学生的身高&#xff08;height&#xff09;... 方法一&#xff1a;利用实例方法来控制 class Student:def __init__(self,name"",age0):self.…

Linux:kubernetes(k8s)prestop事件的使用(10)

他的作用是在结束pod容器之后进行的操作 apiVersion: v1 # api文档版本 kind: Pod # 资源对象类型 metadata: # pod相关的元数据&#xff0c;用于描述pod的数据name: nginx-po # pod名称labels: # pod的标签type: app #这个是随便写的 自定义的标签version: 1.0.0 #这个…

PostgreSQL 安装部署

文章目录 一、PostgreSQL部署方式1.Yum方式部署2.RPM方式部署3.源码方式部署4.二进制方式部署5.Docker方式部署 二、PostgreSQL部署1.Yum方式部署1.1.部署数据库1.2.连接数据库 2.RPM方式部署2.1.部署数据库2.2.连接数据库 3.源码方式部署3.1.准备工作3.2.编译安装3.3.配置数据…

C语言 --- 指针(5)

目录 一.sizeof和strlen对比 1.sizeof 2.strlen 3.strlen 和sizeof的对比 二.数组和指针笔试题目详解 回顾&#xff1a;数组名的理解 1.一维数组 2.字符数组 代码1&#xff1a; 代码2&#xff1a; 代码3&#xff1a; 代码4&#xff1a; 代码5&#xff1a; 代码6&am…

SSM整合项目(使用Vue3 + Element-Plus创建项目基础页面)

1.配置Vue启动端口 1.修改vue.config.js const {defineConfig} require(vue/cli-service) module.exports defineConfig({transpileDependencies: true }) module.exports {devServer: {port: 9999 //启动端口} }2.启动 2.安装Element Plus 命令行输入 npm install eleme…

【leetcode热题】排序链表

给你链表的头结点 head &#xff0c;请将其按 升序 排列并返回 排序后的链表 。 示例 1&#xff1a; 输入&#xff1a;head [4,2,1,3] 输出&#xff1a;[1,2,3,4]示例 2&#xff1a; 输入&#xff1a;head [-1,5,3,4,0] 输出&#xff1a;[-1,0,3,4,5]示例 3&#xff1a; 输入…

骑行瓦恭村的记忆

昆富公路是一条蜿蜒的丝带&#xff0c;引领着校长骑行这群热爱骑行的骑友&#xff0c;穿越时光的隧道&#xff0c;直达心灵深处那个被岁月抚摸得斑斑点点的小村庄——瓦恭村。大普吉隧道如同时光的门扉&#xff0c;每一次穿越&#xff0c;都伴随着轰鸣的排气扇声音&#xff0c;…