linux的僵尸进程和孤儿进程及解决方法

1.僵尸进程:父进程还在执行,子进程已经结束,子进程依然占用部分资源,这时的子进程就是僵尸进程。

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

int main()
{
        pid_t pid;
        pid = fork();
        if(pid < 0)
        {
                fprintf(stderr, "fork error %s\n", strerror(errno));
                exit(1);
        }
        else if(pid > 0)//parent process
        {
                while(1)
                {
                        sleep(3);
                }
                printf("pid: %d, ppid: %d\n", getpid(), getppid());
        }
        else
        {
                printf("pid: %d, ppid: %d\n", getpid(), getppid());
                exit(1);
        }
        return 0;
}

执行结果:

[root@localhost process]# ./03process 
pid: 3384, ppid: 3383

这时打开另一个终端查看进程状态:

[root@localhost Desktop]# ps -ef | grep 03process
root      3383  2718  0 08:34 pts/0    00:00:00 ./03process
root      3384  3383  0 08:34 pts/0    00:00:00 [03process] <defunct>
root      5445  3265  1 08:35 pts/1    00:00:00 grep 03process

此时,3383是父进程,3384是子进程(僵尸进程),只能通过杀死父进程来杀死僵尸进程:

[root@localhost Desktop]# kill -9 3383
[root@localhost Desktop]# ps -ef | grep 03process
root      5455  3265  2 08:37 pts/1    00:00:00 grep 03process

杀死父进程后查看执行终端:

[root@localhost process]# ./03process 
pid: 3384, ppid: 3383
Killed

通过wait()函数来处理僵尸进程(最合理的方式):

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

int main()
{
        pid_t pid;
        pid = fork();
        if(pid < 0)
        {
                fprintf(stderr, "fork error %s\n", strerror(errno));
                exit(1);
        }
        else if(pid > 0)//parent process
        {
                wait(0);
                while(1)
                {
                        sleep(3);
                }
                printf("pid: %d, ppid: %d\n", getpid(), getppid());
        }
        else
        {
                printf("pid: %d, ppid: %d\n", getpid(), getppid());
                exit(1);
        }
        return 0;
}

上面程序中的wait(0)会感知子进程一旦执行完毕,就替它回收资源,从而不会产生僵尸进程。


2.孤儿进程:父进程退出,它的子进程还在运行,那么这些子进程就是孤儿进程,孤儿进程被init进程(1号进程)所收养,并由init进程负责他们的退出时清理工作。

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


int main()
{
        pid_t pid;
        pid = fork();
        if(pid < 0)
        {
                fprintf(stderr, "fork error %s\n", strerror(errno));
                exit(1);
        }
        else if(pid > 0)//parent process
        {
                sleep(1);
                printf("pid: %d, ppid: %d\n", getpid(), getppid());
                exit(1);//parent process exit
        }
        else
        {
                sleep(3);
                printf("pid: %d, ppid: %d\n", getpid(), getppid());
        }
        printf("pid: %d, ppid: %d\n", getpid(), getppid());
        return 0;
}

执行结果:

[root@localhost process]# ./03process 
pid: 5573, ppid: 5572
pid: 5572, ppid: 2718
[root@localhost process]# pid: 5573, ppid: 1

可以看到父进程先结束,子进程成为孤儿进程继续执行,主进程退出,孤儿进程交由1号进程,他负责回收孤儿进程。

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Linux 中,僵尸进程孤儿进程都是指与父进程不再有联系的进程,它们通常是由于进程管理不当或程序错误导致的。 **僵尸进程**是已经完成执行任务,但其父进程还没有来得及处理其退出状态的进程。当进程完成执行后,它的退出状态(也称为退出码或终止状态)会被保存在系统中,直到父进程通过 `wait` 或 `waitpid` 等函数来获取该状态。如果父进程没有处理该状态,那么该进程就会成为僵尸进程,占用系统资源。要清理僵尸进程,可以使用 `kill` 命令向其父进程发送 `SIGCHLD` 信号,或者重新编写程序,使其正确处理子进程的退出状态。 **孤儿进程**是指其父进程已经退出或被终止,但其自身仍在运行的进程。孤儿进程会被 `init` 进程(进程号为 `1`)接管,`init` 进程会定期检查系统中是否有孤儿进程,并且将其的父进程设置为 `init` 进程。要避免孤儿进程的产生,可以在父进程退出之前,等待子进程的退出,或者将子进程的父进程设置为 `init` 进程。 可以使用 `ps` 命令来查看系统中的僵尸进程孤儿进程。使用以下命令可以查看所有僵尸进程: ``` ps aux | grep 'Z' ``` 其中,`aux` 选项用于显示所有进程,`grep 'Z'` 用于查找所有状态为 `Z` 的进程,即僵尸进程。 使用以下命令可以查看所有孤儿进程: ``` ps -ejH ``` 其中,`-e` 选项用于显示所有进程,`-j` 选项用于以层次结构的形式显示进程,`-H` 选项用于显示所有孤儿进程

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值