(转) Linux 的僵尸(zombie)进程

 

原文地址:  http://coolshell.cn/articles/656.html

可能很少有人意识到,在一个进程调用了exit之后,该进程 并非马上就消失掉,而是留下一个称为僵尸进程(Zombie)的数据结构。在Linux进程的5种状态中,僵尸进程是非常特殊的一种,它已经放弃了几乎所 有内存空间,没有任何可执行代码,也不能被调度,仅仅在进程列表中保留一个位置,记载该进程的退出状态等信息供其他进程收集,除此之外,僵尸进程不再占有 任何内存空间。

僵尸进程的来由,要追溯到Unix,Unix的设计者们设计这个东西并非是因为闲来无事想装装酷什么的。上面说到,僵尸进程中保存着很多对程序员和系统管理员非常重要的信息,首先,这个进程是怎么死亡的?是正常退出呢,还是出现了错误,还是被其它进程强迫退出的?也就是说,这个程序的退出码是什么?其次,这个进程占用的总系统CPU时间和总用户CPU时间分别是多少?发生页错误的数目和收到信号的数目。这些信息都被存储在僵尸进程中,试想如果没有僵尸进程,进程执行多长我们并不知道,一旦其退出,所有与之相关的信息都立刻都从系统中清除,而如果此时父进程或系统管理员需要用到,就只好干瞪眼了。

 

所以,进程退出后,系统会把该进程的状态变成Zombie,然后给上一定的时间等着父进程来收集其退出信息,因为可能父进程正忙于别的事情来不及收集,所以,使用Zombie状态表示进程退出了,正在等待父进程收集信息中。

Zombie进程不可以用kill命令清楚,因为进程已退出,如果需要清除这样的进程,那么需要清除其父进程,或是等很长的时间后被内核清除。因为Zombie的进程还占着个进程ID号呢,这样的进程如果很多的话,不利于系统的进程调度。

下面,让我们来看看一个示例:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
/* zombie.c */
#include <sys/types.h>
#include <unistd.h>  main()
{
     pid_t pid; 
     pid=fork();
     if (pid<0) { /* 如果出错 */ 
         printf ( "error occurred!\n" );
     } else if (pid==0){ /* 如果是子进程 */ 
         exit (0);
     } else /* 如果是父进程 */ 
         sleep(60);  /* 休眠60秒 */ 
         wait(NULL); /* 收集僵尸进程 */
     }
}

编译这个程序:

1
$ cc zombie.c -o zombie

后台运行程序,以使我们能够执行下一条命令

1
2
$ . /zombie &
[1] 1217

列一下系统内的进程

1
2
3
4
5
6
$ ps -ax
... ...
1137   pts /0   S   0:00   - bash
1217   pts /0   S   0:00   . /zombie
1218   pts /0   Z   0:00   [zombie]
1578   pts /0   R   0:00   ps   -ax

其中的”Z”就是僵尸进程的标志,它表示1218号进程现在就是一个僵尸进程。

收集Zombie进程的信息,并终结这些僵尸进程,需要我们在父进程中使用waitpid调用和wait调用。这两者的作用都是收集僵尸进程留下的信息,同时使这个进程彻底消失。

 

个人总结:

 

1 僵尸进程的用途

存储进程占用的总系统CPU时间和总用户CPU时间分别是多少?

发生页错误的数目和收到信号的数目.进程退出的状态, 

供回收者查看

 

2 什么是僵尸进程

僵尸进程不是真正的进程, 它是一种状态, 进程一退出就成为了僵尸进程,

只有三种方法才能回收僵尸进程

(1) 父进程显示地调用wait和waitpid回收之

(2) 杀死父进程, 僵尸进程就会过继给 init, init 始终会负责清理僵尸进程

(3) 重启系统

所以,  如果父进程不调用 wait 和 waitpid 回收僵尸进程, 在父进程活着的情况下, 

僵尸进程将一直存在下去

注意, 不能通过 kill  发信号退出, 因为僵尸进程已退出, 无法接受信号

 

3 僵尸进程的危害

这样的进程如果很多的话,塞满process table, 不利于系统的进程调度

 

4  区别僵尸进程与孤儿进程

如果父进程先于子进程退出,  子进程就成为了孤儿进程, 没有父亲的孩子

统统都送到孤儿院 init 进程领养.

孤儿进程的特点是: 父亲死了, 我还活着, 我是孤儿

僵尸进程的特点是: 我死了, 父亲还活着, 但它不给我收尸

 

5 如何防止僵尸进程的产生

(1) 阻塞方式:  父进程显示的调用 wait 或 waitpid 回收僵尸进程

(2) 异步方式:  父进程捕捉 SIGCHLD 信号, 然后调用 wait 或 waitpid 回收僵尸进程

 

6 正确地回收僵尸进程。

如果父进程很忙,就采用异步回收的方式,捕捉 SIGCHLD 信号,其处理函数为:

void sig_chld(int signo)
{
    pid_t pid;
    int stat;

    pid = wait(&stat);
    printf("child %d terminated\n", pid);
}
 

这里存在一个问题,如果多个子进程同时退出,同时产生SIGCHLD信号,

而SIGCHLD是不可靠信号,不支持排队。所以可能只会捕捉到一次,在这种

情况下,只有一个子进程被回收,其它子进程将变成僵尸进程。

 

解决办法是收到SIGCHLD信号后,不是只回收一个子进程,而是回收所有退出了的子进程。

考虑在信号处理函数内循环调用wait, 当没有子进程退出时,wait会一直阻塞,信号处理函数

将无法返回。我们需要的是 waitpid,通过设置 WNOHANG 选项,在没有子进程退出时,

waitpid 返回 -1, 这时 退出循环,信号处理函数返回。

 

void sig_chld(int signo)
{
    pid_t pid;
    int stat;

    while ((pid = waitpid(-1, &stat, WNOHANG)) > 0) {
        printf("child %d terminated\n", pid);
    }
}
 

 

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 在 Linux 中,进程可能是僵尸进程或僵死进程僵尸进程是一种已经结束运行但还没有被父进程回收的进程。当父进程没有调用 wait 或 waitpid 函数来回收子进程的结束状态时,子进程就会成为僵尸进程僵尸进程在系统中仍占用一些资源,但是它的代码已经不会再被执行。 僵死进程是一种无法再终止的进程。当一个进程的父进程已经终止,而这个进程没有被重新分配给其他的父进程时,它就成为了僵死进程。僵死进程不会占用系统资源,但是它仍然占用着进程表中的一个项目。 ### 回答2: 在Linux中,进程既可以是僵尸进程zombie process)也可以是僵死进程(dead process)。 僵尸进程是指一个子进程已经结束运行,但是父进程还没有处理该子进程的终止状态信息。这种状态下的子进程会被内核保留,并在进程表中记录一段时间。僵尸进程不占用系统的资源,但是会占用一个进程ID(PID),因此如果有大量僵尸进程存在,可能会导致系统PID耗尽。处理僵尸进程的常见方法是父进程调用wait或waitpid函数来处理子进程的状态,释放其资源,并将其从进程表中删除。 僵死进程是指一个进程已经终止运行并且无法正常释放资源。这可能是因为进程在结束时未正确释放资源的问题,或者由于系统故障导致进程被异常终止。僵死进程会一直占用系统的资源,包括内存和文件描述符等。要解决僵死进程,可以通过重启系统来清除它们,或者通过手动终止占用资源的进程来释放资源。 因此,Linux中既存在僵尸进程又存在僵死进程僵尸进程是指子进程结束但父进程未处理的进程状态信息,而僵死进程是指已经结束但无法正常释放资源的进程。处理僵尸进程需要父进程主动回收子进程,而对于僵死进程通常需要进行系统重启或手动终止进程来解决。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值