Linux之僵尸进程


* 僵尸进程
子进程结束,父进程没有正确处理子进程返回信息。
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

相关内容