Linux僵尸进程是什么,如何产生又该怎么清理

来源:Vuejs社区作者:卡拉米头衔:草根站长
导读:本期聚焦于小伙伴创作的《Linux僵尸进程是什么,如何产生又该怎么清理》,敬请观看详情,探索知识的价值。以下视频、文章将为您系统阐述其核心内容与价值。如果您觉得《Linux僵尸进程是什么,如何产生又该怎么清理》有用,将其分享出去将是对创作者最好的鼓励。

Linux僵尸进程是进程状态中的一种特殊存在,指的是子进程已经执行结束退出,但是父进程没有调用wait或者waitpid等函数回收子进程的资源,导致子进程的进程描述符仍然保留在系统进程表中的进程,这类进程的进程状态标记为Z,也就是zombie状态。

Linux僵尸进程是什么,如何产生又该怎么清理

僵尸进程的核心特点

僵尸进程本身不会占用系统的CPU资源和内存资源,因为它已经完成了自身的执行逻辑,只是进程表中残留了对应的条目。但是它占用的进程号如果一直不释放,当系统中僵尸进程数量过多时,会导致系统无法分配新的进程号,进而影响新进程的创建。

我们可以通过ps命令查看系统中的进程状态,僵尸进程的状态列会显示为Z,同时进程名通常会标注为<defunct>,示例如下:

# 查看所有进程,过滤出状态为Z的僵尸进程
ps aux | grep -E 'Z|defunct'
# 输出示例
# user     12345  0.0  0.0      0     0 pts/0    Z    10:00   0:00 [test_proc] <defunct>

僵尸进程的产生原因

僵尸进程的产生和Linux的进程回收机制直接相关,核心流程是:

  • 父进程创建子进程,子进程开始执行任务
  • 子进程执行完成,或者因为某些信号退出,此时子进程的状态变为僵尸状态
  • 父进程没有及时处理子进程的退出信号,没有调用wait类函数回收子进程的资源,子进程就会一直保持僵尸状态

我们可以通过一段简单的C语言代码模拟僵尸进程的产生:

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

int main() {
    pid_t pid = fork(); // 创建子进程
    if (pid < 0) {
        perror("fork failed");
        return 1;
    } else if (pid == 0) {
        // 子进程逻辑,执行完直接退出
        printf("子进程PID: %d,即将退出n", getpid());
        exit(0);
    } else {
        // 父进程逻辑,不调用wait回收子进程,休眠30秒
        printf("父进程PID: %d,子进程PID: %dn", getpid(), pid);
        sleep(30);
    }
    return 0;
}

运行上述代码后,在父进程休眠的30秒内,子进程已经退出,父进程没有回收,此时就能通过ps命令看到对应的僵尸进程。

僵尸进程的清理方法

1. 手动清理方式

僵尸进程无法直接通过kill命令杀死,因为它已经处于退出状态,kill命令对这类进程无效。正确的清理方式是找到僵尸进程的父进程,然后向父进程发送SIGCHLD信号,或者终止父进程,让父进程触发资源回收逻辑。

查找僵尸进程父进程的命令如下:

# 查看僵尸进程的父进程PID,PPID就是父进程号
ps -eo pid,ppid,stat,cmd | grep 'Z'

找到父进程PID后,先尝试给父进程发送SIGCHLD信号,让父进程主动回收子进程:

# 假设父进程PID是1234,发送SIGCHLD信号
kill -SIGCHLD 1234

如果发送信号后僵尸进程仍然存在,可以尝试终止父进程,此时子进程会被init进程(PID为1的进程)接管,init进程会自动回收这些僵尸进程的资源:

# 终止父进程,注意如果是重要服务进程需要先确认影响再操作
kill 1234
# 如果普通kill无法终止,可以使用kill -9强制终止
kill -9 1234

2. 自动避免僵尸进程的方法

在编写多进程程序时,可以通过以下方式避免产生僵尸进程:

  • 父进程中调用wait或者waitpid函数,主动回收退出的子进程资源
  • 父进程中注册SIGCHLD信号的处理函数,在信号处理函数中调用waitpid回收子进程
  • 两次fork创建子进程,让子进程的子进程执行任务,子进程退出后,子进程的子进程会被init进程接管,自动完成回收

以下是使用SIGCHLD信号处理函数避免僵尸进程的示例代码:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <signal.h>
#include <sys/wait.h>

// SIGCHLD信号处理函数,回收退出的子进程
void handle_sigchld(int sig) {
    // 循环调用waitpid,回收所有退出的子进程,WNOHANG表示非阻塞
    while (waitpid(-1, NULL, WNOHANG) > 0);
}

int main() {
    struct sigaction sa;
    sa.sa_handler = handle_sigchld;
    sigemptyset(&sa.sa_mask);
    sa.sa_flags = 0;
    // 注册SIGCHLD信号的处理函数
    sigaction(SIGCHLD, &sa, NULL);

    pid_t pid = fork();
    if (pid < 0) {
        perror("fork failed");
        return 1;
    } else if (pid == 0) {
        printf("子进程PID: %d,执行完成退出n", getpid());
        exit(0);
    } else {
        printf("父进程PID: %d,等待子进程退出n", getpid());
        sleep(10); // 父进程休眠,此时子进程退出后会触发信号处理函数回收
    }
    return 0;
}

常见问题说明

很多用户会疑惑僵尸进程会不会占用系统内存,其实僵尸进程已经释放了大部分资源,仅保留了进程号、退出状态等少量信息在进程表中,不会占用实际的内存和CPU资源,只有当数量过多导致进程号耗尽时才会影响系统。

另外,init进程(PID为1)会定期回收接管过来的僵尸进程,所以如果僵尸进程的父进程是init进程,不需要手动处理,等待init自动回收即可。

Linux僵尸进程zombie_process进程清理进程状态修改时间:2026-07-01 09:00:35

免责声明:​ 已尽一切努力确保本网站所含信息的准确性。网站内容多为原创整理与精心编撰,观点力求客观中立。本站旨在免费分享,内容仅供个人学习、研究或参考使用。若引用了第三方作品,版权归原作者所有。如内容涉及您的权益,请联系我们处理。
内容垂直聚焦
专注技术核心技术栏目,确保每篇文章深度聚焦于实用技能。从代码技巧到架构设计,为用户提供无干扰的纯技术知识沉淀,精准满足专业提升需求。
知识结构清晰
覆盖从开发到部署的全链路。AI、前端、编程、数据库、服务器、建站、系统层层递进,构建清晰学习路径,帮助用户系统化掌握开发与运维所需的核心技术。
深度技术解析
拒绝泛泛而谈,深入技术细节与实践难点。无论是数据库优化还是服务器配置,均结合真实场景与代码示例进行剖析,致力于提供可直接应用于工作的解决方案。
专业领域覆盖
精准对应开发生命周期。从前端界面到后端编程,从数据库操作到服务器运维,形成完整闭环,一站式满足全栈工程师和运维人员的技术需求。
即学即用高效
内容强调实操性,步骤清晰、代码完整。用户可根据教程直接复现和应用于自身项目,显著缩短从学习到实践的距离,快速解决开发中的具体问题。
持续更新保障
专注既定技术方向进行长期、稳定的内容输出。确保各栏目技术文章持续更新迭代,紧跟主流技术发展趋势,为用户提供经久不衰的学习价值。