说到僵死进程大家应该经常听过可能优点不明白这个意思,首先僵死进程就是指子进程退出时,父进程并未对其发出的SIGCHLD信号进行适当处理,导致子进程停留在僵死状态等待其父进程为其收尸,这个状态下的子进程就是僵死进程。

就是一个进程在他结束生命时并不是真正意义上的销毁,而是调用了exit()把一个正常的进程变成了僵死进程,这个僵死进程不占有内存,也不会执行代码,更不能被调用,他只是在进程列表中占了个地位而已。

所以解决办法就是:

1.他只需要父进程调用wait()函数来替他收尸然后就完整的结束这一生。否则会一直保存这个僵死的状态。

2.把他父进程给kill了,这样他就变成了一个孤儿进程,父亲没了没人替他收拾,这时候僵死进程就会被过继给一个名叫init()进程,这个进程会给他收尸。

那么如何查看僵死进程?

利用命令 ps

进程后跟着<defunct>的就是僵死进程

下面给大家看看产生僵死进程的情况。

代码:

#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<signal.h>
#include<assert.h>

int main()
{
   
    pid_t pid=fork();//创建子进程
    assert(pid!=-1);
    if(pid==0)//子进程
    {
        printf("child start\n");
        sleep(5);
        printf("child end\n");
    }
    else//父进程
    {
        printf("father start\n");
        sleep(10);//为了使僵死进程出现所以父进程晚结束
        printf("father end\n");

    }
}

让我们的代码运行

java进程 被killed java进程僵死_僵尸进程

以下是代码部分:

上述的代码是父进程自动结束从而导致子进程“解脱”了,但父进程不一定会结束的,所以这时候我们需要父进程调用wait()来给子进程收尸。

#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<signal.h>
#include<assert.h>
void fun(int sign)
{
    pid_t pid=wait(NULL);
    printf("fun=%d\n",pid);//打印子进程到pid

}
int main()
{
    signal(SIGCHLD,fun);//子进程结束时会给父进程发送SIGCHLD这个信号,接受到以后父进程调用fun函数,在fun函数中调用wait()给子进程收尸。
    pid_t pid=fork();//创建子进程
    assert(pid!=-1);
    if(pid==0)//子进程
    {
        printf("child start\n");
        sleep(5);
        printf("child end\n");
    }
    else//父进程
    {
        printf("father start\n");
        sleep(10);//为了使僵死进程出现所以父进程晚结束
        sleep(3);//为了给大家截图看状态,后面会有讲解
        printf("father end\n");

    }
}

 

这是执行截图,大家发现子进程结束后就直接在进程表中消失了

java进程 被killed java进程僵死_僵死进程的解决_02

大家看代码知道父进程中睡眠10s,子进程中睡眠5s,5s后子进程结束会给父进程发SIGCHLD信号,父进程接受到信号会被唤醒执行fun()方法调用wait()给子进程收尸,那大家是不是有个问题,父进程本来睡10s但是睡了5s被子进程打断,唤醒去调用了fun(),那不是还有5s的睡眠时间吗?要不要把剩下的5s睡完,答案是不需要,因为运行过程就是执行代码的过程,sleep(10)这行代码已经执行过了,至于效果那就是另一回事,所以父进程下一步做的是执行下一行代码sleep(3),为了给大家截上面的图反映看父子进程的状态,所以让他睡眠3s来打印出此时子进程还在不在进程表中。

现在大家应该理解僵死进程的产生以及解决了吧