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

僵尸进程的核心特点
僵尸进程本身不会占用系统的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