zoukankan      html  css  js  c++  java
  • linux 如何清理僵尸进程

    今天在维护服务器的时候,发现有5个nova-novncproxy的僵尸进程。
    26327 ?        S      0:05  \_ /usr/bin/python /usr/bin/nova-novncproxy --config-file=/etc/nova/nova.conf
     4765 ?        Z      0:00      \_ [nova-novncproxy] <defunct>
     4766 ?        Z      0:00      \_ [nova-novncproxy] <defunct>
     4767 ?        Z      0:00      \_ [nova-novncproxy] <defunct>
     4768 ?        Z      0:00      \_ [nova-novncproxy] <defunct>
     4769 ?        Z      0:00      \_ [nova-novncproxy] <defunct>
    之前对于僵尸进程的了解并不深,赶紧找了篇相关文章来学习一下,该如何处理。
    定义
    In UNIX System terminology, a process that has terminated,but whose parent has not yet waited for it, is called a zombie.
         在UNIX 系统中,一个进程结束了,但是他的父进程没有等待(调用wait / waitpid)他, 那么他将变成一个僵尸进程.  在fork()/execve()过程中,假设子进程结束时父进程仍存在,而父进程fork()之前既没安装SIGCHLD信号处理函数调用 waitpid()等待子进程结束,又没有显式忽略该信号,则子进程成为僵尸进程。
    如何查看linux系统上的僵尸进程,如何统计有多少僵尸进程?
    #ps -ef | grep defunct
    或者查找状态为Z的进程,Z就是代表zombie process,僵尸进程的意思。
    另外使用top命令查看时有一栏为S,如果状态为Z说明它就是僵尸进程。
    Tasks:  95 total,   1 running,  94 sleeping,   0 stopped,   0 zombie
    top命令中也统计了僵尸进程。或者使用下面的命令:
    ps -ef | grep defunct | grep -v grep | wc -l
    如何杀死僵尸进程呢?
    1.改写父进程,在子进程死后要为它收尸。具体做法是接管SIGCHLD信号。子进程死后,会发送SIGCHLD信号给父进程,父进程收到此信 号后,执行waitpid()函数为子进程收尸。这是基于这样的原理:就算父进程没有调用wait,内核也会向它发送SIGCHLD消息,尽管对的默认处 理是忽略,如果想响应这个消息,可以设置一个处理函数。
      SIGCHLD信号:子进程结束时, 父进程会收到这个信号。如果父进程没有处理这个信号,也没有等待(wait)子进程,子进程虽然终止,但是还会在内核进程表中占有表项,这时的子进程称为 僵尸进程。这种情况我们应该避免(父进程或者忽略SIGCHILD信号,或者捕捉它,或者wait它派生的子进程,或者父进程先终止,这时子进程的终止自 动由init进程来接管)。
      2. kill -18 PPID (PPID是其父进程)
      这个信号是告诉父进程,该子进程已经死亡了,请收回分配给他的资源。
      SIGCONT也是一个有意思的信号。如前所述,当进程停止的时候,这个信号用来告诉进程恢复运行。该信号的有趣的地方在于:它不能被忽略或阻塞,但可以被捕获。缺省行为是丢弃该信号。
      3.终止父进程
      如果方法2不能终止,可采用终止其父进程的方法(如果其父进程不需要的话)父进程死后,僵尸进程成为”孤儿进程”,过继给1号进程init,init始终会负责清理僵尸进程.它产生的所有僵尸进程也跟着消失。
      先看其父进程又无其他子进程,如果有,可能需要先kill其他子进程,也就是兄弟进程。方法是:
      kill –15 PID1 PID2 (PID1,PID2是僵尸进程的父进程的其它子进程)。
      然后再kill父进程:kill –15 PPID
    4.使用脚本终止父进程
         一般僵尸进程很难直接kill掉,不过您可以kill僵尸爸爸。父进程死后,僵尸进程成为”孤儿进程”,过继给1号进程init,init始终会负责清理僵尸进程.它产生的所有僵尸进程也跟着消失。
    ps -e -o ppid,stat | grep Z | cut -d” ” -f2 | xargs kill -9
    kill -HUP `ps -A -ostat,ppid | grep -e ’^[Zz]‘ | awk ’{print $2}’`
    当然您可以自己编写更好的shell脚本,欢迎与大家分享。
    我将nova-novncproxy stop后再start,僵尸进程即消失,问题解决。
    另外子进程死后,会发送SIGCHLD信号给父进程,父进程收到此信号后,执行waitpid()函数为子进程收尸。就是基于这样的原理:就算父进程没有调用wait,内核也会向它发送SIGCHLD消息,而此时,尽管对它的默认处理是忽略,如果想响应这个消息,可以设置一个处理函数。
    如何避免僵尸进程呢?
    处理SIGCHLD信号并不是必须的。但对于某些进程,特别是服务器进程往往在请求到来时生成子进程处理请求。如果父进程不等待子进程结束,子进程将成为僵尸进程(zombie)从而占用系统资源。如果父进程等待子进程结束,将增加父进程的负担,影响服务器进程的并发性能。在Linux下 可以简单地将 SIGCHLD信号的操作设为SIG_IGN。
    signal(SIGCHLD,SIG_IGN);
    这样,内核在子进程结束时不会产生僵尸进程。这一点与BSD4不同,BSD4下必须显式等待子进程结束才能释放僵尸进程
    或者
    用两次fork(),而且使紧跟的子进程直接退出,是的孙子进程成为孤儿进程,从而init进程将负责清除这个孤儿进程。
  • 相关阅读:
    java成神之——ImmutableClass,null检查,字符编码,defaultLogger,可变参数,JavaScriptEngine,2D图,类单例,克隆,修饰符基本操作
    java成神之——注释修饰符
    java成神之——enum枚举操作
    java成神之——集合框架之队列,栈,集合并发
    java成神之——集合框架之Maps,Hashtable
    java成神之——集合框架之ArrayList,Lists,Sets
    java成神之——集合框架之Array
    java成神之——数值操作BigDecimal,BigInteger,Random,SecureRandom
    xmind教程
    linux下部署svn服务器
  • 原文地址:https://www.cnblogs.com/jym-sunshine/p/4859624.html
Copyright © 2011-2022 走看看