首页 > 系统相关 >【Linux探索学习】第十三弹——进程状态:深入理解操作系统进程状态与Linux操作系统中的进程状态

【Linux探索学习】第十三弹——进程状态:深入理解操作系统进程状态与Linux操作系统中的进程状态

时间:2024-11-16 09:19:07浏览次数:3  
标签:状态 TASK 操作系统 Linux 进程 等待

Linux笔记:

https://blog.csdn.net/2301_80220607/category_12805278.html?spm=1001.2014.3001.5482

前言:

在上篇我们已经讲解了进程的基本内容,也了解了进程在操作系统的重要作用,今天我们正式开始进程的另一个知识点的讲解:进程状态,即一个进程不可能一直处在运行或终止状态中,它是可以有很多种状态的,今天我就将详细讲解一下进程具体有哪些状态,以及这些状态之间是如何切换的

目录

一、操作系统中的进程状态概述

1.1 经典的进程状态模型

1.2 进程状态转换图

二、Linux操作系统中的进程状态

2.1 Linux进程状态的分类

2.2 各状态的详细解释

2.3 Linux进程状态表

2.4 使用ps查看进程状态

三、代码示例:实现Linux进程状态的模拟

3.1 阻塞等待状态示例

3.2 僵尸进程示例

四、总结


在操作系统中,进程是资源分配和任务调度的基本单位。为了更好地管理系统资源,操作系统将进程的生命周期划分为不同的状态,帮助系统实现高效的资源调度与管理。Linux操作系统在经典的进程状态模型基础上进行了扩展和细化,提供了更细粒度的控制。本文将详细介绍操作系统进程状态的基本概念,深入解析Linux操作系统中的进程状态,并通过代码示例展示各个状态的实现方式。

一、操作系统中的进程状态概述

操作系统中的进程状态是进程在生命周期中可能处于的不同状态。这些状态帮助操作系统识别进程的运行情况,并在不同状态间进行合理的资源分配。操作系统中的经典进程状态包括就绪、运行、阻塞和终止。

1.1 经典的进程状态模型

典型的操作系统中,进程状态可以分为以下几种:

状态描述
就绪(Ready)进程已准备好,等待CPU调度器分配CPU。
运行(Running)进程正在占用CPU,正在执行中。
阻塞(Blocked/Waiting)进程因需要等待资源或事件而无法继续执行,暂停等待。
终止(Terminated)进程执行完成或被强制终止,进入终止状态,等待操作系统回收资源。

在这种经典模型中,进程会在不同状态之间流转。具体来说:

  • 就绪 -> 运行:CPU调度器分配CPU给就绪状态的进程。
  • 运行 -> 阻塞:进程等待I/O或其他资源时进入阻塞状态。
  • 阻塞 -> 就绪:等待的条件满足后,进程重新进入就绪队列,等待CPU调度。
  • 运行 -> 终止:进程执行完毕或异常终止,进入终止状态。

1.2 进程状态转换图

该图展示了一个经典进程状态的转换流程。箭头表示进程状态转换的可能路径。

二、Linux操作系统中的进程状态

Linux操作系统在经典的进程状态基础上进行了一系列扩展,允许内核更细粒度地控制进程,尤其是当系统资源紧张或多任务并发性很高时。Linux内核中的进程状态可以使用ps命令或读取/proc文件系统来查看进程的状态信息。

2.1 Linux进程状态的分类

Linux状态描述
TASK_RUNNING正在运行或可以运行的进程,即就绪状态。
TASK_INTERRUPTIBLE可中断等待状态,等待期间可以被信号中断并唤醒。
TASK_UNINTERRUPTIBLE不可中断等待状态,等待期间不会被信号唤醒,通常用于等待设备I/O操作的完成。
TASK_STOPPED进程被暂停,通常因收到SIGSTOP等信号。
TASK_TRACED进程被调试器(如gdb)控制。
EXIT_DEAD进程已结束,即将被回收。
EXIT_ZOMBIE僵尸状态,进程结束但资源尚未被父进程回收。
TASK_DEAD进程已彻底终止,资源已回收。

在进程的task_struct结构体中,state字段用来表示进程的当前状态。根据内核中的定义,不同状态的进程会被挂载在不同的等待队列上,以实现细粒度的调度与控制。

2.2 各状态的详细解释

  1. TASK_RUNNING:进程正在运行或准备运行。它可以被调度器分配到CPU执行。TASK_RUNNING的进程始终是就绪队列的一部分。

  2. TASK_INTERRUPTIBLE:进程处于可中断等待状态,等待某一条件满足。进程在此状态下可以被信号唤醒。(关于信号的知识我们在后面会讲)

  3. TASK_UNINTERRUPTIBLE:进程处于不可中断的等待状态。这通常用于等待特定资源(例如设备I/O操作完成)。进程在此状态下不会被信号打断,即便外部发送信号也不会响应。

  4. TASK_STOPPED:进程被暂停,通常是由于接收到SIGSTOP信号或调试器干预,等待继续或恢复信号。

  5. TASK_TRACED:进程被调试工具(如gdb)跟踪和控制。此状态下的进程会暂停,直到调试器进一步控制。

  6. EXIT_DEAD:进程终止后进入清理阶段,等待系统回收资源。

  7. EXIT_ZOMBIE:进程已结束,系统未回收其资源。僵尸进程会保留在系统中,直到其父进程调用wait()系统调用收集它的退出状态。

  8. TASK_DEAD:表示进程已彻底结束,系统已回收其所有资源。

2.3 Linux进程状态表

Linux 状态符号进程描述
TASK_RUNNINGR进程正在运行或可以被调度器调度运行。
TASK_INTERRUPTIBLES进程处于可中断的睡眠状态,等待条件满足。
TASK_UNINTERRUPTIBLED进程处于不可中断的睡眠状态,通常等待I/O。
TASK_STOPPEDT进程暂停或已终止。
EXIT_ZOMBIEZ僵尸进程,等待父进程收回。

2.4 使用ps查看进程状态

在Linux系统中,可以通过ps命令查看进程的状态:

ps -aux

ps命令会显示每个进程的详细信息,其中状态列标记着每个进程的状态。状态的含义如下:

  • R:运行或就绪状态。
  • S:可中断等待。
  • D:不可中断等待。
  • T:停止。
  • Z:僵尸状态。

我们也可以通过ps命令查看某个指定的进程的信息:

ps axj | grep 进程名字

三、代码示例:实现Linux进程状态的模拟

为了更好地理解Linux中的进程状态,以下示例演示了不同状态的应用。代码展示了一个简单的内核模块,用于模拟进程进入阻塞状态、唤醒状态等。

3.1 阻塞等待状态示例

#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/init.h>
#include <linux/sched.h>
#include <linux/wait.h>
#include <linux/kthread.h>
#include <linux/delay.h>

static wait_queue_head_t wait_queue;  // 定义等待队列
static int condition = 0;
static struct task_struct *thread_waiter;

static int waiter_thread(void *data) {
    printk(KERN_INFO "等待线程启动...\n");
    wait_event_interruptible(wait_queue, condition != 0);  // 可中断等待
    printk(KERN_INFO "等待线程被唤醒,继续执行...\n");
    return 0;
}

static int __init wait_queue_init(void) {
    init_waitqueue_head(&wait_queue);  // 初始化等待队列
    thread_waiter = kthread_run(waiter_thread, NULL, "waiter_thread");
    if (IS_ERR(thread_waiter)) {
        printk(KERN_ERR "线程创建失败\n");
        return PTR_ERR(thread_waiter);
    }
    msleep(5000);  // 模拟一些操作
    condition = 1;  // 修改条件
    wake_up_interruptible(&wait_queue);  // 唤醒等待线程
    return 0;
}

static void __exit wait_queue_exit(void) {
    printk(KERN_INFO "卸载模块\n");
}

module_init(wait_queue_init);
module_exit(wait_queue_exit);
MODULE_LICENSE("GPL");

在上面的代码中:

  • waiter_thread进入可中断等待状态,等待条件满足。
  • 当条件变量condition改变时,wake_up_interruptible唤醒等待队列中的线程,使waiter_thread进入可运行状态。

3.2 僵尸进程示例

僵尸进程是进程状态的特殊情况,无法直接控制其生成。为了生成僵尸进程,可以在一个子进程中完成工作后立刻终止,但不等待父进程收回资源。示例如下:

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

int main() {
    pid_t pid = fork();
    if (pid == 0) {
        // 子进程,立即退出成为僵尸进程
        printf("子进程:PID=%d\n", getpid());
        exit(0);
    } else {
        // 父进程,等待5秒以查看僵尸进程
        printf("父进程:PID=%d\n", getpid());
        sleep(5);
        printf("父进程结束,僵尸子进程将被回收。\n");
    }
    return 0;
}

在这个代码中,子进程在终止后不被父进程立即回收,因此变成僵尸状态。可以使用ps命令查看该僵尸进程,发现它的状态为Z

3.3 孤儿进程示例

除了上面的内容外,我们还有一个叫做孤儿进程的特殊进程,比如当fork执行出来的父子进程,父进程先被杀掉的,我们就会发现子进程的PPID会变为1,进程1为systemd,这就是操作系统本身——父进程是1号进程——孤儿进程是被操作系统所领养了

#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
int main()
{
 pid_t id = fork();
 if(id < 0){
 perror("fork");
 return 1;
 }
 else if(id == 0){//child
 printf("I am child, pid : %d\n", getpid());
 sleep(10);
 }else{//parent
 printf("I am parent, pid: %d\n", getpid());
 sleep(3);
 exit(0);
 }
 return 0;
}

四、总结

本文深入讲解了操作系统中经典的进程状态模型,以及Linux系统中对进程状态的细化与扩展。在Linux中,进程状态包括TASK_RUNNINGTASK_INTERRUPTIBLETASK_UNINTERRUPTIBLE等多种状态,为操作系统内核调度提供了更强的控制力。通过本文的理论分析和代码示例,相信大家对进程状态的理解会更加深入。

希望本篇内容能够帮助大家在实际开发和系统调试中更加高效地管理和控制Linux进程状态。


感谢各位大佬观看,创作不易,还请各位大佬点赞支持!!!

标签:状态,TASK,操作系统,Linux,进程,等待
From: https://blog.csdn.net/2301_80220607/article/details/143665231

相关文章

  • Linux系统-redis集群、nacos、nginx、keepalived、mysql开机自启
    一、Redis集群开机自启:如三主三从交叉式redis集群,有两个方法,自行选择。方法一:第一步:分别在各节点添加以下redis.service文件命令:vim/lib/systemd/system/redis_6379.service添加:[Unit]Description=Redispersistentkey-valuedatabaseAfter=network.targetAfter=......
  • 【Linux】系统编程的初探与遐想
    Linux==LinuxisnotUnix.前言 这是我的第一篇关于Linux系统编程的笔记,后面我将会把后续的笔记开源分享到博客。Linux系统介绍系统的特点 1.开源的,即源代码是公开的,大部分是免费的。2.命令行+图形界面的交互形式。 3.多用户操作系统。开源与闭源 1.......
  • 【深入浅出】之Linux多进程实现shell外壳程序(简易版)
    ......
  • 初识Linux · 信号保存
    目录前言:Blockpendinghandler表信号保存前言:前文我们已经介绍了信号产生,在时间的学习线上,信号的学习分为预备知识,信号产生,信号保存,信号处理,本文我们学习信号保存,在前言部分,我们介绍几个信号保存中的概念。信号递达:实际执行信号的处理动作。信号未决:信号从产生到递达......
  • linux下编译安装memcached
    Memcached在Linux下的编译安装一、安装依赖库Memcached依赖于一些系统库,在大多数Linux发行版中,需要安装libevent库。Debian/Ubuntu系统使用以下命令安装依赖库:sudoapt-yupdatesudoapt-yinstalllibevent-devCentOS/RHEL系统可以通过以下命令安装:sudoyum......
  • Linux cp和mv命令 对于目录复制到目录的情况 的 所有情况示例
    cp和mv命令的行为总结表假设以下路径设置:源路径:/nihao或/nihao//nihao包含文件和子目录:file1,dir1/,file2目标路径:/nima或/nima/表格cp命令行为命令目标路径存在?最终路径结构cp-r/nihao/nima/是/nima/nihao/cp-r/nihao//nima/是/ni......
  • linux运维面试题【基础篇】
    前言本篇主要分享linux运维面试过程中常见的面试题材,当时自己面试的时候就遇到3道原题,最终也是顺利通过面试,下面给大家分享一下:面试题库1、描述Linux系统的启动过程电源BIOS自检读取MBR,运行grub加载内核内核启动/sbin/init程序,进行系统初始化根据系统运行级别执行......
  • Linux常用命令之find命令详解
    find命令详解find是Unix和类Unix操作系统(如Linux和macOS)中一个非常强大的命令行工具,用于在文件系统中搜索文件和目录。find命令可以根据多种条件(如文件名、类型、大小、修改时间等)进行搜索,并可以执行复杂的操作。基本语法find命令的基本语法如下:find[路径]......
  • Linux编程:基于 Unix Domain Socket 的进程/线程间通信实时性优化
    文章目录0.引言1.使用`epoll`边缘触发模式非不要不选择阻塞模式边缘触发(ET)模式优点示例2.使用实时调度策略3.CPU绑定4.使用无锁缓冲区5.优化消息传递的大小和频率6.使用`SO_RCVTIMEO`和`SO_SNDTIMEO`7.示例代码其他阅读0.引言前几天被问到“如何优......
  • 32. 线程、进程与协程
    一、什么是多任务  如果一个操作系统上同时运行了多个程序,那么称这个操作系统就是多任务的操作系统,例如:Windows、Mac、Android、IOS、Harmony等。如果是一个程序,它可以同时执行多个事情,那么就称为多任务的程序。  一个CPU默认可以执行一个程序,如果想要多个程序一起执行......