您的位置:首页 > 运维架构 > Linux

Linux下的僵尸进程及避免方法

2016-10-03 11:03 676 查看
僵尸进程就是已经结束了的却还没从进程表中删除的进程。毋庸置疑,僵尸进程过多会导致进程表里面条目满了,从而导致系统奔溃,并不占用系统资源。

僵尸进程很特殊,因为在进程的状态中,它放弃了几乎所有内存空间,没有任何可执行代码,也不可以被调度,只在进程列表里中保留一个位置,记载该进程的退出状态等信息,以便其他进程收集。

产生的原因:每个Linux进程在进程表里都有一个进入点,核心程序执行该进程时使用到的一切信息都存储在进入点。当使用ps命令察看系统中的进程信息时,看到的就是进程表中的相关数据。当以fork()系统调用建立一个新的进程后,核心进程就会在进程表中给这个新进程分配一个进入点,然后将相关信息存储在该进入点所对应的进程表内,这些信息中有一项是其父进程的识别码。当这个进程走完了自己的生命周期后,它会执行exit()系统调用,此时原来进程表中的数据会被该进程的退出码、执行时所用的CPU时间等数据所取代,这些数据会一直保留到系统将它传递给它的父进程为止。进程出现时间是在子进程终止后,但是父进程尚未读取这些数据之前。

可以用top命令查看僵尸进程,最后有defunct标记的,就是僵尸进程。

那么如何避免这些僵尸进程呢?

父进程通过wait和waitpid等函数等待子进程结束,这会导致敷进程挂起。如果父进程很忙,则可以用signal函数为SIGNCHLD安装handler,因为子进程结束后,符进程会收到该信号,可以在handler中调用wait回收。如果父进程不关心子进程什么时候结束,那么可以用“signal(SIGCHLD,SIG_IGN)”通知内核,自己对子进程的结束不感兴趣,那么子进程结束后,内核会回收,并不再给父进程发送信号。

另外,可以fork两次,父进程fork一个子进程,然后继续工作,子进程fork一个孙进程后退出,那么孙进程被init接管,孙进程结束后,init会回收,不过子进程的回收还要自力更生。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息