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

一、僵尸进程的产生原因

我们知道,每个进程在进程表里都有一个进入点(entry),核心程序执行该进程时使用到的一切信息都存储在进入点。当用ps命令察看系统中的进程信息时,看到的就是进程表中的相关数据。

所以,当一个父进程以fork()系统调用建立一个新的子进程后,核心进程就会在进程表中给这个子进程分配一个进入点,然后将相关信息存储在该进入点所对应的进程表内。这些信息中有一项是其父进程的识别码。

而当这个子进程结束的时候(比如调用exit命令结束),其实他并没有真正的被销毁,而是留下一个称为僵尸进程(Zombie)的数据结构(系统调用exit的作用是使进程退出,但是也仅仅限于一个正常的进程变成了一个僵尸进程,并不能完全将其销毁)。此时原来进程表中的数据会被该进程的退出码(exit code)、执行时所用的CPU时间等数据所取代,这些数据会一直保留到系统将它传递给它的父进程为止。由此可见,defunct进程的出现时间是在子进程终止后,但是父进程尚未读取这些数据之前。

此时,该僵尸子进程已经放弃了几乎所有的内存空间,没有任何可执行代码,也不能被调度,仅仅在进程列表中保留一个位置,记载该进程的退出状态信息供其他进程收集,除此之外,僵尸进程不再占有任何存储空间。他需要他的父进程来为他收尸,如果他的父进程没有安装SIGCHLD信号处理函数调用wait 或 waitpid() 等待子进程结束,也没有显式忽略该信号,那么它就一直保持僵尸状态,如果这时候父进程结束了,那么init进程会自动接手这个子进程,为他收尸,他还是能被清除掉的。但是如果父进程是一个循环,不会结束,那么子进程就会一直保持僵尸状态,这就是系统中为什么有时候会有很多的僵尸进程。

二、如何杀死僵尸进程

如上可知,僵尸进程一旦出现之后,很难自己消亡,会一直存在下去,直至系统重启。虽然僵尸进程几乎不占系统资源,但是,这样下去,数量太多了之后,终究会给系统带来其他的影响。因此,如果一旦见到僵尸进程,我们就要将其杀掉。如何杀掉僵尸进程呢?

有同学可能会说,很简单嘛,直接使用kill命令就好啊。或者,实在不行,加一个-9的后缀(kill -9),肯定杀掉!

请注意:defunct状态下的僵尸进程是不能直接使用kill -9命令杀掉的,否则就不叫僵尸进程了。那么,该如何杀呢?

方法有二:

  1. 重启服务器电脑,这个是最简单,最易用的方法,但是如果你服务器电脑上运行有其他的程序,那么这个方法,代价很大。所以,尽量使用下面一种方法。
  2. 找到该defunct僵尸进程的父进程,将该进程的父进程杀掉,则此defunct进程将自动消失。

问题又来了,如何找到defunct僵尸进程的父进程呢?

很简单,一句命令就够了:ps -ef | grep defunct_process_pid。

三、如何预防僵尸进程

以上介绍的只是在发现了僵尸进程之后,如何去杀死它。那么,有同学可能会说了,这个是治标不治本的。真正的办法是,不让它产生,问题才能彻底解决。OK,那我们就来介绍一下,如何预防僵尸进程的产生。

  • 在父进程创建子进程之前,就向系统申明自己并不会对这个子进程的exit动作进行任何关注行为,这样的话,子进程一旦退出后,系统就不会去等待父进程的操作,而是直接将该子进程的资源回收掉,也就不会出现僵尸进程了。具体的办法就是,在父进程的初始化函数中,调用这个函数:signal(SIGCHLD,SIG_IGN);
  • 如果上述语句没来得及调用,也有另外一个办法。那就是在创建完子进程后,用waitpid等待子进程返回,也能达到上述效果;
  • 如果上述两个办法都不愿意采用,那还有一招:在父进程创建子进程的时候,连续调用两次fork(),而且使紧跟的子进程直接退出,使其孙子进程成为孤儿进程,从而init进程将代替父进程来接手,负责清除这个孤儿进程。于是,父进程就无需进行任何的清理行为,系统会自动处理;

本人在实际项目中的具体代码,参考如下:

/**************************************************************************************************
**  函数名称:  start_proc
**  功能描述:  启动进程开始运行
**  输入参数:  无
**  输出参数:  无
**  返回参数:  启动成功返回1,启动失败则返回0
**  注意事项:  这里要fork两次,利用系统孤儿进程的回收机制来处理。否则,会出现僵尸进程。具体做法是
**  注意事项:  利用一代子进程再产生二代子进程,同时将一代子进程结束掉,并在父进程中进行收尸处理
**************************************************************************************************/
int start_proc_by_name(const char* procname)
{pid_t first_pid, second_pid;char  filename[100];sprintf(filename, "%s%s", PROC_FILE_PATH, procname);printf("going to start proc by filename \"%s\"...\n", filename);if (access(filename, X_OK | F_OK) != 0) {                                  /* 确认文件属性是否正确 */printf("access filename \"%s\" fail!!!\n", filename);return 0;}first_pid = fork();                                                        /* 创建一代子进程 */if (first_pid < 0) {                                                       /* 一代子进程创建失败 */printf("fork 1st child_proc fail!!!\n");return 0;} else if (first_pid == 0) {                                               /* 创建成功,此处是一代子进程的代码 */printf("fork 1st child_proc success. this is 1st child_proc going to fork 2nd child_proc...\n");second_pid = fork();                                                   /* 创建二代孙进程 */if (second_pid < 0) {                                                  /* 二代孙进程创建失败 */printf("fork 2nd child_proc fail!!!\n");return 0;} else if (second_pid == 0) {                                          /* 创建成功,此处是二代孙进程的代码 */printf("fork 2nd child_proc success. this is 2nd child_proc running...\n");if (execl(filename, procname, (char *)NULL) != -1) {               /* 在子进程中执行该程序 */printf("2nd child_proc: execl proc \"%s\" success!\n", procname);return 1;                                                      /* 执行完毕直接退出 */} else {printf("2nd child_proc: execl proc \"%s\" fail!\n", procname);return 0;}} else {                                                               /* 创建成功,此处是一代子进程的代码 */printf("fork 2nd child_proc success. 1st child_proc exiting...\n");sleep(1);exit(1);}} else {                                                                   /* 创建成功,此处是父进程的代码 */printf("fork 1st child_proc success. father_proc continue to run...\n");if (waitpid(first_pid, NULL, 0) != first_pid) {                        /* 父进程必须为一代子进程收尸 */printf("fork 1st child_proc success. 1st child_proc killed...\n");}}return 1;
}

怎么样,很简单吧?不妨去试试吧……

linux下的僵尸进程产生原因和解决方法(含具体代码)相关推荐

  1. Linux的僵尸进程产生原因及解决方法

    来源:http://www.blogdaren.com/post-882.html 1. 产生原因: 在UNIX 系统中,一个进程结束了,但是他的父进程没有等待(调用wait / waitpid)他, ...

  2. linux下的僵尸进程处理SIGCHLD信号

    什么是僵尸进程? 首先内核会释放终止进程(调用了exit系统调用)所使用的所有存储区,关闭所有打开的文件等,但内核为每一个终止子进程保存了一定量的信息.这些信息至少包括进程ID,进程的终止状态,以及该 ...

  3. linux 检查僵死进程,Linux下杀僵尸进程办法

    Linux认证辅导:Linux下杀僵尸进程办法 为了方便广大考生更好的复习,小编编辑整理提供了Linux认证:Linux下杀僵尸进程办法,以供各位考生考试复习参考,希望对考生复习有所帮助. 1) 检查 ...

  4. linux服务器出现黄,linux服务器出现严重故障后的原因以及解决方法

    linux服务器出现严重故障后的原因以及解决方法 发布时间:2011-11-24 16:32:18   作者:佚名   我要评论 linux服务器出现严重故障后的解决方法,本文为大家介绍四个步骤解决l ...

  5. java进程消失_Linux系统下的Java进程无故消失的解决方法

    很多用户表示,Linux系统下的一些Java项目总是会无缘无故的消失,这是为什么呢?小编认为这极有可能是被系统自动清除多余进程,或是其他程序关掉了Java项目.针对此问题,下面,U大侠小编就给大家介绍 ...

  6. win7系统提示计算机内存不足,Win7系统下提示内存不足的原因及解决方法

    很多时候,我们在使用电脑的过程中遇到难题或是故障在所难免的.最近就有很多win7系统的用户表示,电脑系统老是提示"计算机内存不足,如要还原足够的内存以使程序正确工作,请保存文件,然后关闭或重 ...

  7. Linux下is not in the sudoers file解决方法

    Linux下is not in the sudoers file解决方法 当我们使用sudo命令切换用户的时候可能会遇到提示以下错误:用户名 is not in the sudoers file. 原 ...

  8. Linux下安装mysql后无法启动的解决方法

    在Linux下安装完mysql后,mysql服务无法启动,总是failer.       这个不是因为mysql安装失败,而是因为启动了SELinux.       进入/etc/selinux/co ...

  9. /var/spool/clientmqueue目录下存在大量文件的原因及解决方法

    /var/spool/clientmqueue目录下存在大量文件的原因及解决方法 参考文章: (1)/var/spool/clientmqueue目录下存在大量文件的原因及解决方法 (2)https: ...

最新文章

  1. 【知识发现】python开源哈夫曼编码库huffman
  2. 笔记-知识产权与标准化知识-GB/T-12504-1990计算机软件质量保证计划规范
  3. Android中使用Adapter(适配器)给RecycleView设置数据源
  4. html点击按钮弹出悬浮窗_网课助手1.0 支持悬浮窗搜题等多功能
  5. pythongoogle.probuf.timestamp_gRPC快速入门(一)——Protobuf简介
  6. goquery php,golang:Goquery简单爬虫实例
  7. SCO UNIX环境下自动增加网关的两种方法
  8. Mp4v2实现h264+aac打包成Mp4视频文件
  9. Java网络编程之流的详解
  10. Android框架揭秘-Android服务概要笔记
  11. zemax操作例子_光学软件使用实例:从Zemax导入光学系统
  12. 单文件小型数据库的选择
  13. SeaweedFS安装部署
  14. 【转载】用Pwnage + Redsnow 制作完美越狱固件
  15. 三种嵌入式操作系统的深入分析与比较
  16. 深入理解SLAM中的Marginalization
  17. (二)ElasticSearch实战基础教程(ElasticSearch入门)
  18. 戴尔U2520DR型号显示器连接MacbookPro突然不亮了
  19. java判断车牌号,包含新能源
  20. java毕业设计家庭理财记账系统(附源码、数据库)

热门文章

  1. 如何配置本地yum源
  2. Linux命令行初体验
  3. 2020,那些惊艳我的产品迭代
  4. java中的对象是什么_Java中的对象是什么以及如何使用它?
  5. OpenCV~图像处理API(逆光、模糊、亮度、雾霾)
  6. Javascript实现扫雷游戏
  7. GitHub Actions 漏洞可导致攻击者投毒开发管道
  8. 一位老先生对年轻人不得不说的十大忠告〔转〕
  9. leetcode-回文字符串(马拉车算法模板)
  10. 求sin(x)的近似值