92.Linux的僵死进程以及处理方法

目录

1.什么是僵死进程?

2.代码演示僵死进程 

3.解决办法


1.什么是僵死进程?

  1. 僵死进程是指一个子进程在父进程之前结束,但父进程没有正确地等待(使用 waitwaitpid 等系统调用)来获取子进程的退出状态。当一个进程结束时,它的退出状态会一直保存,直到父进程通过适当的系统调用来获取它。如果父进程没有获取子进程的退出状态,那么子进程就会变成僵死状态。
  2. 我们的程序在退出的时候:return 0,exit(0),之类的,这个0就是退出码,状态信息,这个东西存储在当前进程的PCB中,会有一个整型值来存储退出码。
  3. 当我们子进程结束以后,会把退出码写到PCB中,然后希望父进程可以获得到这个退出码,然后父进程就可以看到子进程是正常运行结束还是出错退出。正常的话我们return 0,失败的话我们return -1。

2.代码演示僵死进程 

#include 
#include 
#include 
#include 
#include 

int main(int argc, char *argv[], char *envp)
{
    char *s = NULL;
    int n = 0;
    pid_t pid = fork();
    assert(pid != -1);
    if (pid == 0)
    {
        s = "child";
        n = 4;
    }
    else
    {
        s = "parent";
        n = 10;
    }
    int i = 0;
    for (; i < n; i++)
    {
        printf("pid=%d,s=%s\n", getpid(), s);
    }

    exit(0);
}

92.Linux的僵死进程以及处理方法_第1张图片 

 从上图中可以看到,当子进程结束后,并没有消失,仍然可以在系统中观测到,但此时子进程其实已经运行结束了,此时子进程的状态被称为僵死状态,系统把处于该类状态的进程称为僵死进程< defunct >

3.解决办法

        在 Unix/Linux 系统中,可以使用 waitwaitpid 函数等待子进程的终止,并获取其退出状态。如果父进程没有执行这些调用,子进程就会一直保留在僵死状态,占用系统资源。

#include 
#include 
#include 
#include 
#include 

int main(int argc, char *argv[], char *envp)
{
    char *s = NULL;
    int n = 0;
    pid_t pid = fork();
    assert(pid != -1);
    if (pid == 0)
    {
        s = "child";
        n = 4;
    }
    else
    {
        s = "parent";
        n = 10;

        int val = 0;
        pid_t id = wait(&val);
        if (WIFEXITED(val))
        {
            printf("id=%d,val=%d\n", id, WEXITSTATUS(val));
        }
    }
    int i = 0;
    for (; i < n; i++)
    {
        printf("pid=%d,s=%s\n", getpid(), s);
    }

    exit(0);
}

92.Linux的僵死进程以及处理方法_第2张图片

显而易见,int val = 0; pid_t id = wait(&val); 的执行起到了阻塞作用,因为要获取子进程的退出码,如果子进程还没有结束,就获取不到,父进程就阻塞住了,先将子进程执行完后,获取到退出码了,才开始执行父进程。 

 

你可能感兴趣的:(Linux学习历程,linux,运维,服务器)