百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术分析 > 正文

linux后台开发中避免僵尸进程的方法总结

liebian365 2025-02-07 18:14 12 浏览 0 评论

一、什么是僵死进程?

一般情况下,程序调用exit(包括_exit和_Exit,它们的区别这里不做解释),它的绝大多数内存和相关的资源已经被内核释放掉,但是在进程表中这个进程项(entry)还保留着(进程ID,退出状态,占用的资源等等),你可能会问,为什么这么麻烦,直接释放完资源不就行了吗?这是因为有时它的父进程想了解它的退出状态。在子进程退出但还未被其父进程“收尸”之前,该子进程就是僵死进程,或者僵尸进程。如果父进程先于子进程去世,那么子进程将被init进程收养,这个时候init就是这个子进程的父进程。

所以一旦出现父进程长期运行,而又没有显示调用wait或者waitpid,同时也没有处理SIGCHLD信号,这个时候init进程就没有办法来替子进程收尸,这个时候,子进程就真的成了“僵尸”了。

二、僵死进程与孤儿进程的区别?

回答这个问题很简单,就是爸爸(父进程)和儿子(子进程)谁先死的问题!

如果当儿子还在世的时候,爸爸去世了,那么儿子就成孤儿了,这个时候儿子就会被init收养,换句话说,init进程充当了儿子的爸爸,所以等到儿子去世的时候,就由init进程来为其收尸。

如果当爸爸还活着的时候,儿子死了,这个时候如果爸爸不给儿子收尸,那么儿子就会变成僵尸进程。

三、僵死进程的危害?

  1. 僵死进程的PID还占据着,意味着海量的子进程会占据满进程表项,会使后来的进程无法fork.
  2. 僵死进程的内核栈无法被释放掉(1K 或者 2K大小),为啥会留着它的内核栈,因为在栈的最低端,有着thread_info结构,它包含着 struct_task 结构,这里面包含着一些退出信息。

四、避免僵死进程的方法

网上搜了下,总结有三种方方法:

① 程序中显示的调用signal(SIGCHLD, SIG_IGN)来忽略SIGCHLD信号,这样子进程结束后,由内核来wai和释放资源

② fork两次,第一次fork的子进程在fork完成后直接退出,这样第二次fork得到的子进程就没有爸爸了,它会自动被老祖宗init收养,init会负责释放它的资源,这样就不会有“僵尸”产生了

③ 对子进程进行wait,释放它们的资源,但是父进程一般没工夫在那里守着,等着子进程的退出,所以,一般使用信号的方式来处理,在收到SIGCHLD信号的时候,在信号处理函数中调用wait操作来释放他们的资源。

五、对每个避免僵死进程方法的解析与总结

首先我们让我们来看一个生成僵尸进程的程序zombie.c如下:

#include???
#include???
#include???
??
int?main(int?argc,?const?char?*argv[])??
{??
????int?i;??
????pid_t?pid;??
??
????for?(i?=?0;?i?

运行程序,在10s睡眠期间使用ps查看进程,你会发现有10个标记为“defunct”的僵尸进程:

需要C/C++ Linux服务器架构师学习资料私信“资料”(资料包括C/C++,Linux,golang技术,Nginx,ZeroMQ,MySQL,Redis,fastdfs,MongoDB,ZK,流媒体,CDN,P2P,K8S,Docker,TCP/IP,协程,DPDK,ffmpeg等),免费分享

接下来看第一种方法,程序avoid_zombie1.c如下:

#include???
#include???
#include???
#include???
#include???
??
int?main(int?argc,?const?char?*argv[])??
{??
????pid_t?pid;??
??
????if?(SIG_ERR?==?signal(SIGCHLD,?SIG_IGN))?{??
????????perror("signal?error");??
????????_exit(EXIT_FAILURE);??
????}??
??
????while?(1)?{??
????????if?((pid?=?fork())?==?0)????/*?child?*/??
????????????_exit(0);??
????}??
??
????exit(EXIT_SUCCESS);??
}??

程序运行期间通过ps命令的确没有发现僵尸进程的存在。

在man文档中有这段话:

Note that even though the default disposition of SIGCHLD is "ignore", explicitly setting the disposition to SIG_IGN results in different treatment of zombie process children.

意思是说尽管系统对信号SIGCHLD的默认处理就是“ignore”,但是显示的设置成SIG_IGN的处理方式在在这里会表现不同的处理方式(即子进程结束后,资源由系统自动收回,所以不会产生僵尸进程),这是信号SIGCHLD与其他信号的不同之处。

在man文档中同样有这样一段话:

The original POSIX standard left the behavior of setting SIGCHLD to SIG_IGN unspecified. 看来这个方法不是每个平台都使用,尤其在一些老的系统中,兼容性不是很好,所以如果你在写一个可移植的程序的话,不推荐使用这个方法。

第二种方法,即通过两次fork来避免僵尸进程,我们来看一个例子avoid_zombie2.c:

#include???
#include???
#include???
#include???
#include???
??
int?main(int?argc,?const?char?*argv[])??
{??
????pid_t?pid;??
??
????while?(1)?{??
????????if?((pid?=?fork())?==?0)?{??/*?child?*/??
????????????if?((pid?=?fork())?>?0)??
????????????????_exit(0);??
????????????sleep(1);??
????????????printf("grandchild,?parent?id?=?%ld\n",??
????????????????????????????(long)getppid());??
????????????_exit(0);??
????????}??
????????if?(waitpid(-1,?NULL,?0)?!=?pid)?{??
????????????perror("waitpid?error");??
????????????_exit(EXIT_FAILURE);??
????????}??
????}??
??
????exit(EXIT_SUCCESS);??
}??

这的确是个有效的办法,但是我想这个方法不适宜网络并发服务器中,应为fork的效率是不高的。

最后来看第三种方法, 也是最通用的方法

先看我们的测试程序avoid_zombie3.c

#include???
#include???
#include???
#include???
#include???
#include???
#include???
#include???
#include???
??
??
void?avoid_zombies_handler(int?signo)??
{??
????pid_t?pid;??
????int?exit_status;??
????int?saved_errno?=?errno;??
??
????while?((pid?=?waitpid(-1,?&exit_status,?WNOHANG))?>?0)?{??
????????/*?do?nothing?*/??
????}??
??
????errno?=?saved_errno;??
}??
??
int?main(int?argc,?char?*argv[])??
{??
????pid_t?pid;??
????int?status;??
????struct?sigaction?child_act;???
??
????memset(&child_act,?0,?sizeof(struct?sigaction));??
????child_act.sa_handler?=?avoid_zombies_handler;??
????child_act.sa_flags?=?SA_RESTART?|?SA_NOCLDSTOP;???
????sigemptyset(&child_act.sa_mask);??
????if?(sigaction(SIGCHLD,?&child_act,?NULL)?==?-1)?{??
????????perror("sigaction?error");??
????????_exit(EXIT_FAILURE);??
????}??
??
????while?(1)?{??
????????if?((pid?=?fork())?==?0)?{??/*?child?process?*/??
????????????_exit(0);??
????????}?else?if?(pid?>?0)?{????????/*?parent?process?*/??
????????}??
????}??
??????
????_exit(EXIT_SUCCESS);??
}??

首先需要知道三点:

1. 当某个信号的信号处理函数被调用时,该信号会被操作系统阻塞(默认sa_flags不设置SA_NODEFER标志)。

2.当某个信号的信号处理函数被调用时,该信号阻塞时,该信号又多次发生,那么操作系统并不将它们排队,而是只保留第一次的,后续的被抛弃。

还有一点我们必须清楚的是

3. wait系列函数与信号SIGCHLD是没有任何关系的,即wait系列函数并不是信号SIGCHLD驱动的。

这个时候,肯定有人有疑问了,既然会丢弃信号,那怎么保证可以收回所有的僵尸进程呢?

关于这个问题,我们可以这样来理解,当子进程结束时,不管有没有产生SIGCHLD信号,或者子进程产生了SIGCHLD信号,而不管父进程有没有收到SIGCHLD信号,这都与子进程已经终止这个事实无关,就是说,子进程终止与信号其实没有任何关系,只是操作系统在子进程终止时会发送信号SIGCHLD给父进程,告之其子进程终止的消息,这样的话,父进程就可以做相应的操作了。而wait系列函数的目的就是收回子进程终止时残留在进程列表中的信息,所以任何时候调用while ((pid = waitpid(-1, &exit_status, WNOHANG)) > 0)都可以收回所有的僵尸进程信息(可以参考下面的程序)。但是这里为什么放在信号处理函数中处理了,这样做的原因是:子进程什么时候结束是个异步事件,而信号机制就是用来处理异步事件的,所以当子进程结束时,可以迅速的收回其残余信息,这样系统中就不会积累大量的僵尸进程了。

也可以这样来理解:系统把所有的僵尸进程串在一起形成一个僵尸进程链表,而while ((pid = waitpid(-1, &exit_status, WNOHANG)) > 0)就是来清空这个链表的,直到waitpid()返回0,表明已经没有僵尸进程了,或者返回-1,表明出错(当错误码errno为ECHILD的时候同样表明已经不存在僵尸进程了)。

了解了以上知识点,就能理解为什么while ((pid = waitpid(-1, &exit_status, WNOHANG)) > 0)能够回收所有的僵尸进程了。

我们可以在上面的信号处理函数中加入相应的打印信息:

static?int?num1?=?0??
static?int?num2?=?0;??
void?avoid_zombies_handler(int?signo)??
{??
????pid_t?pid;??
????int?exit_status;??
????int?saved_errno?=?errno;??
??
????printf("num1?=?%d\n",?++num1);??
????while?((pid?=?waitpid(-1,?&exit_status,?WNOHANG))?>?0)?{??
????????printf("num2?=?%d\n",?++num2);??
????}??
??
????errno?=?saved_errno;??
}??

打印的结果你会发现,当num1递增1的时候,即每调用一次信号处理函数,num2一般会递增很多,即while循环了很多次,所以尽管有的SIGCHLD信号被丢弃了,但是我们不用担心子进程的残余信息会收不回来。退出while循环时,证明此时系统中已经没有僵尸进程了,所以退出信号处理函数后,阻塞的唯一SIGCHLD信号会再次触发该信号处理函数,这样我们就不用担心了。我们不防做个最坏的打算,即之前的信号全部被丢弃了,只有最后一次的SIGCHLD信号被捕获,从而触发了信号处理函数,这样我们也不用担心,因为while循环会一次性收回全部的僵尸进程信息,只是这次循环的次数要多得多罢了,当然这只是假设,一般系统不会出现这样的情况(可以参考本文最后一个程序事例)。

为了证明wait系统函数与信号SIGCHLD没有任何关系,我们可以做个简单的实验,代码如下:

#include???
#include???
#include???
#include???
#include???
??
int?main(int?argc,?char?*argv[])??
{??
????int?i;??
????pid_t?pid;??
??
????for?(i?=?0;?i??0)?{??
????????/*?do?nothing?*/??
????}??
????sleep(10);??
??
????_exit(EXIT_SUCCESS);??
}??

以下是打印结果:

可以看到第一次sleep时系统中积累了5个僵尸进程,第二次sleep时,那5个僵尸进程都被收回了。这个也明显的看到了使用信号处理函数的优势,即可以保证系统不会积累大量的僵尸进程,它可以迅速的清理掉系统中的僵尸进程。

相关推荐

4万多吨豪华游轮遇险 竟是因为这个原因……

(观察者网讯)4.7万吨豪华游轮搁浅,竟是因为油量太低?据观察者网此前报道,挪威游轮“维京天空”号上周六(23日)在挪威近海发生引擎故障搁浅。船上载有1300多人,其中28人受伤住院。经过数天的调...

“菜鸟黑客”必用兵器之“渗透测试篇二”

"菜鸟黑客"必用兵器之"渗透测试篇二"上篇文章主要针对伙伴们对"渗透测试"应该如何学习?"渗透测试"的基本流程?本篇文章继续上次的分享,接着介绍一下黑客们常用的渗透测试工具有哪些?以及用实验环境让大家...

科幻春晚丨《震动羽翼说“Hello”》两万年星间飞行,探测器对地球的最终告白

作者|藤井太洋译者|祝力新【编者按】2021年科幻春晚的最后一篇小说,来自大家喜爱的日本科幻作家藤井太洋。小说将视角放在一颗太空探测器上,延续了他一贯的浪漫风格。...

麦子陪你做作业(二):KEGG通路数据库的正确打开姿势

作者:麦子KEGG是通路数据库中最庞大的,涵盖基因组网络信息,主要注释基因的功能和调控关系。当我们选到了合适的候选分子,单变量研究也已做完,接着研究机制的时便可使用到它。你需要了解你的分子目前已有哪些...

知存科技王绍迪:突破存储墙瓶颈,详解存算一体架构优势

智东西(公众号:zhidxcom)编辑|韦世玮智东西6月5日消息,近日,在落幕不久的GTIC2021嵌入式AI创新峰会上,知存科技CEO王绍迪博士以《存算一体AI芯片:AIoT设备的算力新选择》...

每日新闻播报(September 14)_每日新闻播报英文

AnOscarstatuestandscoveredwithplasticduringpreparationsleadinguptothe87thAcademyAward...

香港新巴城巴开放实时到站数据 供科技界研发使用

中新网3月22日电据香港《明报》报道,香港特区政府致力推动智慧城市,鼓励公私营机构开放数据,以便科技界研发使用。香港运输署21日与新巴及城巴(两巴)公司签署谅解备忘录,两巴将于2019年第3季度,开...

5款不容错过的APP: Red Bull Alert,Flipagram,WifiMapper

本周有不少非常出色的app推出,鸵鸟电台做了一个小合集。亮相本周榜单的有WifiMapper's安卓版的app,其中包含了RedBull的一款新型闹钟,还有一款可爱的怪物主题益智游戏。一起来看看我...

Qt动画效果展示_qt显示图片

今天在这篇博文中,主要实践Qt动画,做一个实例来讲解Qt动画使用,其界面如下图所示(由于没有录制为gif动画图片,所以请各位下载查看效果):该程序使用应用程序单窗口,主窗口继承于QMainWindow...

如何从0到1设计实现一门自己的脚本语言

作者:dong...

三年级语文上册 仿写句子 需要的直接下载打印吧

描写秋天的好句好段1.秋天来了,山野变成了美丽的图画。苹果露出红红的脸庞,梨树挂起金黄的灯笼,高粱举起了燃烧的火把。大雁在天空一会儿写“人”字,一会儿写“一”字。2.花园里,菊花争奇斗艳,红的似火,粉...

C++|那些一看就很简洁、优雅、经典的小代码段

目录0等概率随机洗牌:1大小写转换2字符串复制...

二年级上册语文必考句子仿写,家长打印,孩子照着练

二年级上册语文必考句子仿写,家长打印,孩子照着练。具体如下:...

一年级语文上 句子专项练习(可打印)

...

亲自上阵!C++ 大佬深度“剧透”:C++26 将如何在代码生成上对抗 Rust?

...

取消回复欢迎 发表评论: