Linux系统编程——僵尸进程及wait/waitpid函数

僵尸进程

一般来说,当父进程fork()了一个子进程之后,它有义务把子进程回收。但如果子进程运行完毕后,父进程尚未回收,此时的子进程残留信息(PCB)存放于内核中,变成僵尸进程。(若父进程先于子进程结束,则子进程成为孤儿进程。)
我们知道,子进程的结束和父进程的运行是异步的,那么,父进程会不会不知道子进程已经结束了呢?UNIX中存在一种机制,一个进程调用exit()来结束自己的生命,内核释放该进程所有的资源,包括打开的文件,占用的内存等。但还保留了PCB存放于进程列表中,PCB主要记录了进程的退出状态和进程号。当父进程想要知道子进程的退出状态时,就可以随时查看,进而释放PCB。其实,每个进程都会变成僵尸进程。只是,当父进程快速的回收子进程的时候,我们察觉不到,而父进程还没回收的子进程,则残留在内核中成为了真正的僵尸进程。
通过top指令我们可以看到,僵尸进程并不占用内存资源,但这并不意味着它就没有危害。系统的进程号是有限的,如果产生大量的僵尸进程,则会占用大量的进程号,这就会导致新创建的进程无进程号可用。

僵尸进程代码如下:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/wait.h>

int main(void)
{
    pid_t pid;
    pid = fork();

    if (pid == 0)
    {
        printf("-----child, my parent = %d, going to sleep 10s\n", getppid());
        sleep(10);
        printf("-------------child die---------\n");
    }
    else if (pid > 0)
    {
        while (1)
        {
            printf("I am parent, pid = %d, myson = %d\n", getpid(), pid);
            sleep(1);
        }
    }
    else
    {
        perror("fork");
        return -1;
    }

    return 0;
}

运行结果:
这里写图片描述

另外开一个shell,使用ps aux命令查看进程:
这里写图片描述
红框圈起来的就是僵尸进程,<defunct>的意思是死亡。

使用top -p 404 -p 405命令查看僵尸进程的内存占用情况:
这里写图片描述
S是状态栏,Z(zombie)代表僵尸进程,S(sleeping)代表父进程。在虚拟内存(VIRT)和物理内存(RES)一栏,父进程均有占用,而僵尸进程则占用为0。
既然僵尸进程对系统有一定的危害,那么有没有什么办法回收呢?这就要用到我们下面要讲到的wait和waitpid函数。

wait()函数

一个进程在终止时会关闭所有文件描述符,释放在用户空间分配的内存

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值