揅笑晏晏 2014-04-01
* 僵尸进程
子进程结束,父进程没有正确处理子进程返回信息。
PS:直到父进程退出,子进程变为孤儿进程,其父进程会变为Init进程(PPID=0),Init进程会负责清理僵尸进程
* 危害
僵尸进程没有从进程列表删除,占据内核资源
* 结论
多进程编程,父进程需要跟踪子进程的退出状态
* 僵尸进程查看方法:
1 top (zombie数量)
Tasks: 581 total, 1 running, 574 sleeping, 5 stopped, 1 zombie
2 ps -ef (defunct标记)
1022 16123 16122 0 11:32 pts/12 00:00:00 [a.out] <defunct>
* 僵尸进程处理方法:
当子进程结束,其会给父进程发送SIGCHLD信号。
这时父进程捕获信号后可调用wait()或者waitpid()函数回收子进程尸体
1 pid_t wait (int * status);
2 pid_t waitpid(pid_t pid,int * status,int options);
PS:使用waitpid就够了,其为wait提供非阻塞功能
PS:也可简单使用signal(SIGCHLD,SIG_IGN);这时内核在子进程结束不会产生僵尸进程
* 编程例子
#include <iostream>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/wait.h>
using namespace std;
void OnSigChild(int iSig)
{
pid_t iPid;
int iStat;
while ((iPid = waitpid(-1, &iStat, WNOHANG)) > 0)
{
cout << "<OnSigChild>, Pid=" << iPid << endl;
}
}
int main()
{
signal(SIGCHLD, OnSigChild);
if (!fork())
{
// Child
cout << "Child:" << getpid() << endl;
exit(0);
}
// Parent
while(1);
return 0;
}
* 其他补充
PS:子进程未结束,而父进程先结束了,子进程PPID变为0(Init进程)。这时ps -ef子进程并不是僵尸态
gapp_devnet_1:~ # ps -ef | grep 18849
UID PID PPID C STIME TTY TIME CMD
1022 18849 1 0 11:35 pts/12 00:00:00 ./a.out