Linux 进程状态、僵尸进程与孤儿进程

news2024/12/27 18:10:24

目录

0.前言

1. 进程状态

1.1 定义

1.2 常见进程

2.僵尸进程

2.1 定义

2.2 示例

2.3 僵尸进程的危害与防止方法

3. 孤儿进程

3.1 介绍

3.2 示例

4.小结


(图像由AI生成) 

0.前言

在上一篇文章中,我们介绍了进程的基本概念、进程控制块(PCB)以及如何查看 Linux 系统中的进程状态。这一篇,我们将继续深入探讨进程的各种状态,尤其是僵尸进程和孤儿进程等特殊情况,这些知识对于理解 Linux 系统中的进程管理和资源分配至关重要。

1. 进程状态

1.1 定义

在 Linux 中,进程的状态是进程当前所处的活动阶段或状态的指示。这一状态的定义在 Linux 内核源码中用 task_struct 结构体中的 state 字段表示。这个字段是一个位图,表示进程在何种情况下进行等待或执行操作。内核中的进程状态通过常量数组 task_state_array[] 定义,如下所示:

/*
* 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 */
};

通过这个数组,Linux 内核能够为不同状态的进程进行标识,每个状态都有不同的含义:

  • R (running):表示进程正在运行或准备运行。这是一个活动状态,进程在 CPU 上获得执行时间或正在等待被调度。

  • S (sleeping):表示进程正在睡眠,通常是因为它在等待某个事件发生,比如等待输入/输出(I/O)操作的完成。当条件满足时,进程会被唤醒并恢复运行。

  • D (disk sleep):表示进程处于不可中断的睡眠状态,通常是在等待硬件资源(如磁盘 I/O)的响应。这种状态下的进程不会响应任何信号,除非硬件操作完成。

  • T (stopped):表示进程已被暂停,通常是由用户通过发送 SIGSTOPSIGTSTP 信号来停止进程的执行。进程在这种状态下不会继续执行任何操作,直到接收到继续信号(如 SIGCONT)。

  • t (tracing stop):表示进程被调试器跟踪并暂停执行。调试器可以通过此状态检查进程的运行情况,并可以继续、停止或修改进程的执行。

  • X (dead):表示进程已终止并且即将从系统中清除,通常意味着进程已经彻底退出,内存和资源已释放。

  • Z (zombie):表示僵尸进程,即进程已经结束执行,但其父进程还没有通过 wait() 系统调用获取其退出状态。僵尸进程只保留进程控制块(PCB),不再消耗其他系统资源。

1.2 常见进程

在日常使用 Linux 系统时,上述这些状态中的几种是比较常见的。让我们详细介绍其中几种常见的进程状态:

  1. 运行(R (running)
    这是最活跃的状态,进程处于运行中,意味着该进程正在使用 CPU 资源。即使进程等待 CPU 调度,只要它准备好执行,仍会被标记为 R 状态。这也是大多数活跃进程的正常状态。

  2. 可中断睡眠(S (sleeping)
    这是最常见的等待状态,表示进程正在等待某些条件(如 I/O 完成)。当等待条件满足后,进程将自动被唤醒,恢复执行。使用 pstop 命令时,许多系统后台进程经常处于这种状态。

  3. 不可中断睡眠(D (disk sleep)
    进程处于深度睡眠中,通常在等待某些阻塞的硬件操作(例如磁盘 I/O)。这种状态下的进程无法被信号唤醒,必须等到请求的操作完成。此状态比较罕见,出现时通常与系统 I/O 瓶颈或硬件问题相关。

  4. 停止(T (stopped)
    停止状态的进程通常是被用户手动暂停的。通过 Ctrl+Zkill -STOP 可以将进程送入此状态,典型应用场景是在调试时或暂时停止某些前台任务。

  5. 僵尸(Z (zombie)
    僵尸进程虽然很少见,但却是系统管理中需要注意的一种状态。它们不会消耗系统资源,但过多的僵尸进程可能导致进程号耗尽,系统无法再创建新的进程。

2.僵尸进程

2.1 定义

僵尸进程(Zombie Process)是指一个进程已经完成了它的执行(通过调用 exit() 退出),但它的父进程还没有调用 wait() 系统调用获取它的退出状态。此时,子进程的进程控制块(PCB)依然保存在内存中,等待父进程来回收它的退出信息。在 Linux 中,每个进程都需要一个进程号(PID),如果僵尸进程不被处理,系统将保留它们的 PID,导致可用的 PID 资源被耗尽,最终可能影响系统的正常运行。

僵尸进程本身并不会占用过多的系统资源,但如果产生大量僵尸进程,可能会导致系统进程号耗尽,影响新进程的创建。一般情况下,僵尸进程应该尽快通过父进程调用 wait() 系列函数来处理和回收。

2.2 示例

下面我们通过一个简单的 C 语言代码示例来演示如何创建一个僵尸进程。此代码会创建一个子进程,子进程立即退出,但父进程不调用 wait(),从而让子进程进入僵尸状态。

C语言创建僵尸进程的例子

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>

int main() {
    pid_t pid = fork();

    if (pid > 0) {
        // 父进程睡眠一段时间,以便子进程进入僵尸状态
        sleep(30); // 这里父进程故意不调用 wait(),让子进程变成僵尸
    } else if (pid == 0) {
        // 子进程立即退出
        printf("子进程 (PID: %d) 已经退出,但父进程未调用 wait(),它将成为僵尸进程。\n", getpid());
        exit(0);
    } else {
        // fork 失败
        perror("fork");
        exit(1);
    }

    return 0;
}

在这个示例中,父进程 fork() 出一个子进程,然后父进程进入睡眠状态,而子进程立即 exit() 退出。由于父进程没有调用 wait(),子进程的状态将变为僵尸状态,直到父进程被唤醒并调用 wait() 函数处理它。

我们可以通过 Linux 中的命令行脚本实时查看该进程的状态。以下是一个简单的命令行脚本,它会每秒钟打印一次所有进程的状态,以便实时监控僵尸进程:

实时监控进程状态的脚本

#!/bin/bash

# 使用 ps 命令每秒钟查看进程的状态
while true; do
    clear
    ps -eo pid,ppid,stat,cmd | grep '[Zz]'
    sleep 1
done

这个脚本使用 ps -eo pid,ppid,stat,cmd 命令来打印出所有进程的进程号、父进程号、状态和命令。grep '[Zz]' 会过滤出所有处于僵尸状态的进程。脚本会每秒执行一次,显示系统中当前的僵尸进程。运行这个脚本时,用户可以观察到子进程在退出后变成僵尸进程,并在父进程睡眠期间保持僵尸状态。

2.3 僵尸进程的危害与防止方法

僵尸进程的主要危害并不在于它们占用了大量的系统资源,而是:

  1. 占用进程号:每个进程在系统中都有一个唯一的进程号(PID)。系统中的 PID 数量是有限的,过多的僵尸进程会消耗这些可用的进程号,导致系统在需要创建新进程时无法分配新的 PID。如果系统中存在大量僵尸进程,将导致系统无法创建新的进程,从而影响系统的正常运行。

  2. 系统管理复杂化:僵尸进程虽然不占用 CPU 和内存,但它们会增加系统中进程的数量,给系统管理员带来困扰。在 pstop 这样的命令中,这些僵尸进程会持续存在,使得进程管理和问题排查变得更加复杂。

  3. 资源泄露风险:虽然僵尸进程只保留了少量信息,但如果父进程长期不处理这些僵尸进程,相关资源无法完全释放,尤其在一些复杂的进程间通信或多进程的应用中,可能导致资源泄露或系统资源耗尽。

如何防止僵尸进程

  • 父进程主动调用 wait()waitpid():父进程应当及时调用这些函数来处理子进程的退出状态,避免子进程变为僵尸状态。
  • 使用信号处理机制:父进程可以设置 SIGCHLD 信号处理程序,当子进程结束时自动调用 wait() 来清理僵尸进程。
  • init 进程接管:当父进程终止时,孤儿进程会自动被 init 进程接管,init 进程会负责处理这些子进程,避免僵尸状态的出现。

3. 孤儿进程

3.1 介绍

孤儿进程(Orphan Process)是指其父进程已经终止,但子进程仍在继续运行的进程。当父进程退出后,系统自动将这些孤儿进程的父进程重定向为 init 进程(在大多数 Linux 系统中,PID 为 1),由 init 进程负责管理它们。因此,孤儿进程并不会对系统造成问题,init 进程会在这些孤儿进程终止时回收它们的资源。

孤儿进程本身并不是错误或需要纠正的状态,更多情况下,它是一种自然现象。例如,当某个后台服务的父进程意外终止时,子进程继续执行,成为孤儿进程并由 init 进程接管。这种机制确保了系统的稳定性,防止了因为父进程退出而造成的子进程资源泄漏。

3.2 示例

为了更好地理解孤儿进程的工作方式,我们通过一个 C 语言示例代码来演示孤儿进程的生成和处理过程。在此示例中,父进程创建一个子进程,父进程立即终止,而子进程继续运行并成为孤儿进程。

C语言创建孤儿进程的例子

#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 if (pid == 0) {
        // 子进程
        printf("子进程 (PID: %d, PPID: %d) 开始运行\n", getpid(), getppid());
        sleep(5);  // 子进程等待几秒,观察它成为孤儿进程的状态
        printf("子进程 (PID: %d, PPID: %d) 仍然在运行,并已成为孤儿进程\n", getpid(), getppid());
        exit(0);
    } else {
        // fork 失败
        perror("fork");
        exit(1);
    }

    return 0;
}

解释

  1. 父进程通过调用 fork() 创建了一个子进程。
  2. 父进程立即调用 exit() 终止自身,使得子进程成为孤儿进程。
  3. 子进程在父进程退出后继续执行,并等待 5 秒,确保自己成为孤儿进程。
  4. 子进程打印出自己在成为孤儿进程后的状态,父进程的 PID 变为 1,表示 init 进程已经接管了它。

从输出中可以看到,子进程在父进程退出后继续运行,且子进程的父进程 ID (PPID) 变为 1,说明它已经被 init 进程接管。这就是孤儿进程的典型处理过程。

孤儿进程通常不会给系统带来危害,因为 init 进程会自动管理它们。

4.小结

本篇文章我们深入探讨了 Linux 系统中的进程状态,了解了系统如何管理进程的不同阶段。尤其是僵尸进程和孤儿进程这两种特殊进程状态,在日常的系统管理和故障排查中经常遇到。理解这些概念不仅有助于我们优化系统资源的使用,还能够在需要时快速定位和处理系统中潜在的问题。对于僵尸进程,关键在于父进程的正确处理;对于孤儿进程,系统的 init 进程会自动接管,用户通常不需要过多干预。

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

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

相关文章

Python Flask 和 Django 的区别与适用场景

Flask 和 Django 的异同&#xff08;结合代码解释&#xff09; Flask 和 Django 是两个流行的 Python Web 框架。尽管它们都是用于构建 Web 应用程序的强大工具&#xff0c;但它们的设计哲学、功能和用法有很大的区别。通过代码示例&#xff0c;可以更直观地理解 Flask 和 Dja…

基础岛第1关:书生大模型全链路开源体系

了解书生浦语大模型体系&#xff1a;书生浦语 InternLM2.5 系列模型&#xff1a; 卓越的推理性能&#xff1a;在数学推理方面取得了同量级模型最优精度&#xff0c;超越了 Llama3 和 Gemma2-9B。有效支持百万字超长上下文&#xff1a;模型在 1 百万字长输入中几乎完美地实现长…

一文讲透大语言模型构建流程

最近已有不少大厂都在秋招宣讲了&#xff0c;也有一些在 Offer 发放阶段。 节前&#xff0c;我们邀请了一些互联网大厂朋友、今年参加社招和校招面试的同学。 针对新手如何入门算法岗、该如何准备面试攻略、面试常考点、大模型技术趋势、算法项目落地经验分享等热门话题进行了…

用友U8-CRM fillbacksettingedit.php SQL注入复现

0x01 产品描述&#xff1a; 用友U8-CRM是企业利用信息技术&#xff0c;是一项商业策略&#xff0c;它通过依据市场细分组织企业资源、培养以客户为中心的经营行为、执行以客户为中心的业务流程等手段来优化企业的客户满意度和获利能力。 0x02 漏洞描述&#xff1a; 用友 U8 C…

VMware Aria Operations for Logs 8.18 发布,新增功能概览

VMware Aria Operations for Logs 8.18 - 集中式日志管理 请访问原文链接&#xff1a;https://sysin.org/blog/vmware-aria-operations-for-logs/&#xff0c;查看最新版。原创作品&#xff0c;转载请保留出处。 作者主页&#xff1a;sysin.org 集中式日志管理 VMware Aria …

web开发(1)-基础

这是对b站课程的总结&#xff0c;后续可能会继续更 01 前后端分离介绍_哔哩哔哩_bilibili01 前后端分离介绍是Web应用开发-后端基础-基于Springboot框架的第1集视频&#xff0c;该合集共计29集&#xff0c;视频收藏或关注UP主&#xff0c;及时了解更多相关视频内容。https://w…

计算机前沿技术-人工智能算法-大语言模型-最新研究进展-2024-10-01

计算机前沿技术-人工智能算法-大语言模型-最新研究进展-2024-10-01 目录 文章目录 计算机前沿技术-人工智能算法-大语言模型-最新研究进展-2024-10-01目录1. Beyond Text-to-Text: An Overview of Multimodal and Generative Artificial Intelligence for Education Using Topi…

第二弹:面向对象编程中的类与对象

文章目录 面向对象编程中的类与对象1. 类与对象的定义1.1 类和对象的概念1.2 类的基本定义 2. 类的封装2.1 类的封装语法2.2 类成员访问权限2.3 struct和class的区别2.4 类封装与成员函数定义分离 3. 类对象的创建与销毁3.1 静态与动态对象的创建3.2 对象的销毁 4. 构造函数和析…

云服务器部署k8s需要什么配置?

云服务器部署k8s需要什么配置&#xff1f;云服务器部署K8s需要至少2核CPU、4GB内存、50GBSSD存储的主节点用于管理集群&#xff0c;工作节点建议至少2核CPU、2GB内存、20GBSSD。还需安装Docker&#xff0c;选择兼容的Kubernetes版本&#xff0c;配置网络插件&#xff0c;以及确…

大数据-154 Apache Druid 架构与原理详解 基础架构、架构演进

点一下关注吧&#xff01;&#xff01;&#xff01;非常感谢&#xff01;&#xff01;持续更新&#xff01;&#xff01;&#xff01; 目前已经更新到了&#xff1a; Hadoop&#xff08;已更完&#xff09;HDFS&#xff08;已更完&#xff09;MapReduce&#xff08;已更完&am…

【日常记录】现在遇到的Y7000P亮度无法调节问题,无需改动注册表进行调整的方法。

1、winR 2、输入&#xff1a;services.msc 3、找到下面红框内的服务 4、右键后&#xff0c;点击重启任务&#xff0c;重启任务后&#xff0c;再次按热键即可恢复亮度调节。

XGBOOST算法Python实现(保姆级)

摘要 XGBoost算法&#xff08;eXtreme Gradient Boosting&#xff09;在目前的Kaggle、数学建模和大数据应用等竞赛中非常流行。本文将会从XGBOOST算法原理、Python实现、敏感性分析和实际应用进行详细说明。 目录 0 绪论 一、材料准备 二、算法原理 三、算法Python实现 3…

西电25考研 VS 24考研专业课大纲变动汇总

01专业课变动 西安电子科技大学专业课学长看到953网络安全基础综合变为 893网络安全基础综合&#xff0c;这是因为工科要求都必须是8开头的专业课&#xff0c;里面参考课本还是没变的&#xff0c;无非就是变了一个名字 对于其他变动专业课也是同理的 02专业课考纲内容变化 对于…

<<迷雾>> 第5章 从逻辑学到逻辑电路(6)--莎士比亚电路 示例电路

info::操作说明 鼠标单击开关切换开合状态 primary::在线交互操作链接 https://cc.xiaogd.net/?startCircuitLinkhttps://book.xiaogd.net/cyjsjdmw-examples/assets/circuit/cyjsjdmw-ch05-17-shakespeare-circuit.txt 原图

【AI人工智能】文心智能体,陪爸妈去旅游,国庆假期不容错过,旅游搭子首选

文章目录 背景创作灵感陪爸妈去旅游简介角色与目标思考路径个性化开场白调优 智能体体验总结和感受 背景 文心智能体平台&#xff0c;开启新一轮活动&#xff0c;超级创造营持续百日活动。 在AI 浪潮席卷的今天&#xff0c;如雨后春笋般丛生的 AI 应用&#xff0c;昭告着时代风…

AI助力CMIP6数据处理技术及在气候变化、生态农业、水文多领域实践应用

查看原文>>>AI助力CMIP6数据处理技术及在气候变化、生态农业、水文多领域实践应用 目录 专题一 CMIP6中的模式比较计划 专题二 数据下载 专题三 基础知识3.1 Python基础 专题四 单点降尺度 专题五 统计方法的区域降尺度 专题六 基于WRF模式的动力降尺度 专题七…

墙绘艺术在线交易:SpringBoot技术解析

2 相关技术 2.1 SSM框架介绍 本课题程序开发使用到的框架技术&#xff0c;英文名称缩写是SSM&#xff0c;在JavaWeb开发中使用的流行框架有SSH、SSM、SpringMVC等&#xff0c;作为一个课题程序采用SSH框架也可以&#xff0c;SSM框架也可以&#xff0c;SpringMVC也可以。SSH框架…

习题2 算法

选择题 1. 以下叙述中错误的是 【 正确答案: C】。 A.算法正确的程序最终一定会结束 B.算法正确的程序可以有零个输入 C.算法正确的程序可以有零个输出 D.算法正确的程序对于相同的输入一定有相同的结果 2. 下面关于算法的说法错误的是 【 正确答案: B】。 …

推送k8s镜像到阿里云服务器

1、服务打包 2、打包后进入Dockerfile的同级目录 运行 docker build -t 镜像名:镜像版本 . (这个点是当前目录的意思&#xff0c;不能忽略)例如 docker build -t trac:v1.0.4 .3、上传镜像到阿里云镜像服务 注意选择区域 例如&#xff1a; docker tag 70743d9bdba3 registr…

git push 远程仓库 linux版

git push 远程仓库 为了将git本地仓库的内容push到远程仓库上&#xff0c;此处记录一下具体的过程&#xff1a; 1.进入到项目根目录下 2.将本地目录初始化为Git仓库。默认情况下&#xff0c;初始分支为main。 git init && git symbolic-ref HEAD refs/heads/main3.…