僵尸进程条目就从进程表中删除,执行waitpid()函

2019-12-01 13:25 来源:未知

永利平台娱乐 1

ps        --forest        ASCII art process tree

 

2 怎样来清除僵尸进程: 

永利平台娱乐 2

  1.改写父进程,在子进程死后要为它收尸。具体做法是接管SIGCHLD信号。子进程死后,会发送SIGCHLD信号给父进程,父进程收到此信号后,执行waitpid()函数为子进程收尸。这是基于这样的原理:就算父进程没有调用wait,内核也会向它发送SIGCHLD消息,尽管对的默认处理是忽略,如果想响应这个消息,可以设置一个处理函数。 

 

  2.把父进程杀掉。父进程死后,僵尸进程成为"孤儿进程",过继给1号进程init,init始终会负责清理僵尸进程.它产生的所有僵尸进程也跟着消失。

 

3 僵尸进程的危害:

在UNIX系统中,僵尸进程是指完成执行(通过exit系统调用,或运行时发生致命错误或收到终止信号所致)但在操作系统的进程表中仍然有一个表项(进程控制块PCB),处于”终止状态“的进程。这发生于子进程需要保留表项以允许其父进程读取子进程的exit status:一旦退出态通过wait系统调用读取,僵尸进程条目就从进程表中删除,称之为”回收(reaped)”。正常情况下,进程直接被其父进程wait并由系统回收。进程长时间保持僵尸状态一般是错误的并导致资源泄漏。

在Unix系统管理中,当用ps命令观察进程的执行状态时,经常看到某些进程的状态栏为defunct,这就是所谓的“僵尸”进程。“僵尸”进程是一个早已死亡的进程,但在进程表(processs table)中仍占了一个位置(slot)。由于进程表的容量是有限的,所以,defunct进程不仅占用系统的内存资源,影响系统的性能,而且如果其数目太多,还会导致系统瘫痪。

英文术语zombie process源自en:zombie — 不死之人,隐喻子进程已死但仍然没有被收割。与正常进程不同,[kill](http://man.wuguiyunwei.com/index.php/2017/06/01/1097.html)命令对僵尸进程无效。孤儿进程不同于僵尸进程,其父进程已经死掉,但孤儿进程仍能正常执行,但并不会变为僵尸进程,因为被init(进程ID号为1)收养并wait其退出。

 

子进程死后,系统会发送SIGCHLD 信号给父进程,父进程对其默认处理是忽略。如果想响应这个消息,父进程通常在SIGCHLD 信号事件处理程序中,使用``wait系统调用来响应子进程的终止。

<1>现象。

僵尸进程被收割后,其进程号(PID)与在进程表中的表项都可以被系统重用。但如果父进程没有调用wait,僵尸进程将保留进程表中的表项,导致了资源泄漏。某些情况下这反倒是期望的:父进程创建了另外一个子进程,并希望具有不同的进程号。如果父进程通过设置事件处理函数为SIG_IGN显式忽略SIGCHLD信号,而不是隐式默认忽略该信号,或者具有SA_NOCLDWAIT标志,所有子进程的退出状态信息将被抛弃并且直接被系统回收。

i249 ~ # top
top - 13:35:49 up 738 days, 1:46, 3 users, load average: 3.91, 4.64, 4.94
Tasks: 175 total, 1 running, 168 sleeping, 0 stopped, 6 zombie
Cpu0 : 93.8%us, 0.0%sy, 0.0%ni, 6.2%id, 0.0%wa, 0.0%hi, 0.0%si, 0.0%st
Cpu1 :100.0%us, 0.0%sy, 0.0%ni, 0.0%id, 0.0%wa, 0.0%hi, 0.0%si, 0.0%st
Cpu2 : 21.4%us, 0.0%sy, 0.0%ni, 78.6%id, 0.0%wa, 0.0%hi, 0.0%si, 0.0%st
Cpu3 : 23.5%us, 5.9%sy, 0.0%ni, 64.7%id, 0.0%wa, 0.0%hi, 5.9%si, 0.0%st
Mem: 16385312k total, 15502200k used, 883112k free, 947972k buffers
Swap: 33574908k total, 0k used, 33574908k free, 9233612k cached

UNIX命令`ps列出的进程的状态(”STAT”)栏标示为 “Z`“则为僵尸进程。

PID USER PR NI VIRT RES SHR S %CPU %MEM TIME+ COMMAND
7854 uuwatch 20 0 2496m 1.2g 11m S 216 7.8 48377:55 java
31880 uuwatch 20 0 3993m 2.8g 12m S 39 18.1 1013:10 java
13279 root 20 0 10696 1152 796 R 7 0.0 0:00.03 top

收割僵尸进程的方法是通过kill命令手工向其父进程发送SIGCHLD信号。如果其父进程仍然拒绝收割僵尸进程,则终止父进程,使得init进程收养僵尸进程。init进程周期执行wait系统调用收割其收养的所有僵尸进程。

<2>查找。

为避免产生僵尸进程,实际应用中一般采取的方式是:

i249 ~ # ps -A -o stat,ppid,pid,cmd |grep -e "^[Zz]"
Zs 12798 12800 [runscript.sh] <defunct>
Zs 12896 12897 [runscript.sh] <defunct>
Zs 12964 12965 [runscript.sh] <defunct>
Zs 13060 13061 [runscript.sh] <defunct>
Zs 13126 13127 [runscript.sh] <defunct>
Zs 13199 13200 [runscript.sh] <defunct>

  1. 将父进程中对SIGCHLD信号的处理函数设为SIG_IGN(忽略信号);
  2. fork两次并杀死一级子进程,令二级子进程成为孤儿进程而被init所“收养”、清理。

<3>解决。

例子:

 

 

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
#include &lt;sys/wait.h&gt;
#include &lt;stdlib.h&gt;
#include &lt;unistd.h&gt;
 
int main(void)
{
  pid_t pids[10];
  int i;
 
  for (i = 9; i &gt;= 0; --i) {
    pids[i] = fork();
    if (pids[i] == 0) {
      sleep(i+1);
      _exit(0);
    }
  }
 
  for (i = 9; i &gt;= 0; --i)
    waitpid(pids[i], NULL, 0);
 
  return 0;
}

我们的公共号

永利平台娱乐 3

 

i249 ~ # ps -A -o stat,ppid,pid,cmd |grep -e "^[Zz]"|awk '{print $2}'|xargs kill -9

 

进程状态。
            D=不可中断的睡眠状态
            R=运行
            S永利平台娱乐,=睡眠
            T=跟踪/停止
            Z=僵尸进程

TAG标签:
版权声明:本文由永利平台娱乐发布于IT交流,转载请注明出处:僵尸进程条目就从进程表中删除,执行waitpid()函