避免产生僵尸进程的N种方法(zombie process)
http://blog.csdn.net/duyiwuer2009/article/details/7964795
认识僵尸进程
1、如果父进程先退出
子进程自动被 init 进程收养,不会产生僵尸进程
2、如果子进程先退出
2.1 父进程 wait() 处理,则僵尸进程会被父进程清理
2.2 如果父进程不用 wait() 处理,则僵尸进程会在父进程退出之前一直存在。当然,父进程退出后,僵尸子进程会被 init 收养,init 进程会自动调用 wait() 处理。但是对于处理网络请求的服务器进程来说,父进程可能会一直存在,子进程处理完任务就退出,这种情况下会产生很多僵尸进程,这种场景就需要对僵尸进程的处理提高警惕了。
避免产生僵尸进程的5种方法
1、推荐方法:fock twice, 用孙子进程去完成子进程的任务(注意这种方法的使用情景)(http://blog.csdn.net/duyiwuer2009/article/details/7948040)
2、wait(), 但是会使父进程阻塞
3、signal(SIGCHLD,SIG_IGN), 并不是所有系统都兼容
4、sigaction + SA_NOCLDWAIT, 并不是所有系统都兼容
5、推荐方法:在signal handler中调用 waitpid (下面的例子能说明用 waitpid 而不用 wait的原因),这样父进程不用阻塞
注意:每一种方法都有它适用的场合,比如方法 5 适用于 one-request-one-process 的网络服务器程序,而方法 1 则不适合。
关于对处理SIGCHLD或SIGCLD的讨论,APUE 10.7(http://infohost.nmt.edu/~eweiss/222_book/222_book.html) 是最权威最经典的,不过可能由于书出得较早,考虑了太多老系统,当时的系统对信号的处理还不是很完善,使得书中对这个问题的讨论显得相当复杂,但对于现在的系统,我们可以简化许多。
方法4的代码:
#include <signal.h>
#include <stdlib.h>
#include <stdio.h>
#include <unistd.h>
int main()
{
pid_t pid;
struct sigaction sa;
/* prevent zombies */
sa.sa_handler = SIG_IGN;
sa.sa_flags = SA_NOCLDWAIT;
)
{
perror("sigaction");
exit();
}
)
{
exit();
}
)
{
sleep();
exit();
}
sleep();
;
}
方法5的代码:
/**
* Final (correct) version of sig_chld function that calls waitpid.
*
* UNIX Network Programming Volume 1, Third Edition, 5.9(Handling 'SIGCHLD' Signals) and 5.10
*/
void sig_chld(int signo)
{
pid_t pid;
int stat;
, &stat, WNOHANG)) > )
{
printf("child %d terminated\n", pid);
}
return;
}
【参考资料】
UNIX Network Programming Volume 1, Third Edition, 5.9(Handling 'SIGCHLD' Signals) and 5.10
http://topic.csdn.net/u/20090911/16/5860C371-DBF9-440A-851B-C6AD26B6E480.html
http://baike.baidu.com/view/758736.htm
线程和进程的分离,http://blog.chinaunix.net/space.php?uid=317451&do=blog&id=92626
How do I get rid of zombie processes that persevere, http://www.faqs.org/faqs/unix-faq/faq/part3/section-13.html
http://www.ccur.com/isdfaq/How_do_I_avoid_creating_zombies.txt
避免产生僵尸进程的N种方法(zombie process)的更多相关文章
- linux下避免僵尸进程的几种方法
linux下我们可以调用fork函数创建子进程,创建的子进程将会得到父进程的数据空间.堆.栈......副本(采用写时复制机制),子进程将会继承父进程的信号掩码.信号处理方式.当前工作目录.会话id. ...
- Linux启动新进程的三种方法
程序中,我们有时需要启动一个新的进程,来完成其他的工作.下面介绍了三种实现方法,以及这三种方法之间的区别. 1.system函数-调用shell进程,开启新进程system函数,是通过启动shell进 ...
- 查找linux系统下的端口被占用进程的两种方法 【转】
在linux下开发时,你的软件可能要使用某一个端口,或者想查找某一个端口是否被占用.需要怎么做呢??这的确是一个比较烦恼的问题,我也此为这个苦恼过.但是通过查找man手册,还是同事的交流.总结出来两种 ...
- linux 中获取进程和kill进程的几种方法
ps: ps命令是最基本同时也是非常强大的进程查看命令,使用该命令可以确定有哪些进程正在运行和运行的状态.进程是否结束.进程有没有僵尸.哪些进程占用了过多的资源等等. 注意:ps是显示瞬间进程的状态, ...
- Linux启动新进程的几种方法汇总
有时候,我们需要在自己的程序(进程)中启动另一个程序(进程)来帮助我们完成一些工作,那么我们需要怎么才能在自己的进程中启动其他的进程呢?在Linux中提供了不少的方法来实现这一点,下面就来介绍一个这些 ...
- Linux启动新进程的几种方法及比较
有时候,我们需要在自己的程序(进程)中启动另一个程序(进程)来帮助我们完成一些工作,那么我们需要怎么才能在自己的进程中启动其他的进程呢?在Linux中提供了不少的方法来实现这一点,下面就来介绍一个这些 ...
- Linux启动新进程的几种方法及比较[转]
有时候,我们需要在自己的程序(进程)中启动另一个程序(进程)来帮助我们完成一些工作,那么我们需要怎么才能在自己的进程中启动其他的进程呢?在Linux中提供了不少的方法来实现这一点,下面就来介绍一个这些 ...
- Linux中创建Daemon进程的三种方法
什么是daemon进程? Unix/Linux中的daemon进程类似于Windows中的后台服务进程,一直在后台运行运行,例如http服务进程nginx,ssh服务进程sshd等.注意,其英文拼写为 ...
- Linux查看进程的4种方法
转至:https://www.cnblogs.com/hml-blog-com/p/11558369.html 进程是在 CPU 及内存中运行的程序代码,而每个进程可以创建一个或多个进程(父子进程). ...
随机推荐
- Effective Java 读书笔记之十 序列化
一.谨慎地实现Serializable接口 1.一旦一个类被发布,就大大地降低了“改变这个类的实现”的灵活性. 2.仔细设计类的序列化形式而不是接受类的默认虚拟化形式. 3.反序列化机制是一个“隐藏的 ...
- Android隐藏标题栏和状态栏
一.隐藏标题栏 //隐藏标题栏 this.requestWindowFeature(Window.FEATURE_NO_TITLE); 二.隐藏状态栏 //隐藏状态栏 this.getWindow() ...
- Android Studio日志工具的使用
Android Studio的LogCat工具 Verbose:对应Log.v(),这个方法用于打印那些最为琐碎的信息,意义最小的日志信息.是Android日志里面级别最低的一种. Debug:对应L ...
- RPC(Remote Procedure Call Protocol)——远程过程调用协议 学习总结
首先了解什么叫RPC,为什么要RPC,RPC是指远程过程调用,也就是说两台服务器A,B,一个应用部署在A服务器上,想要调用B服务器上应用提供的函数/方法,由于不在一个内存空间,不能直接调用,需 ...
- Apache shutdown unexpectedly启动错误解决方法
这个问题比较常见, 通常是80.443端口被占用 cmd 通过运行apache/bin/httpd.exe 打印如下log: (OS 10048)通常每个套接字地址(协议/网络地址/端口)只允许使用一 ...
- python gui之tkinter语法杂记
随手写的,后续整理. 1. Entry如何调用父类的__init__? 以下错,原因后续分析 super(Your_Entry_Class, self).__init__(self,**kw) 报错: ...
- 【leetcode】Maximum Gap
Maximum Gap Given an unsorted array, find the maximum difference between the successive elements in ...
- java获取对象属性类型、属性名称、属性值
/** * 根据属性名获取属性值 * */ private Object getFieldValueByName(String fieldName, Object o) { try { String ...
- C++ 中超类化和子类化常用API
在windows平台上,使用C++实现子类化和超类化常用的API并不多,由于这些API函数的详解和使用方法,网上一大把.本文仅作为笔记,简单的记录一下. 子类化:SetWindowLong,GetWi ...
- Java使用for循环打印乘法口诀(正倒左右三角形)
代码1: public void test1(){ for(int i = 1; i < 10 ; i ++){ for(int k = 1; k < i ; k ++){ System. ...