僵尸进程,孤儿进程

Z(zombie)——僵尸进程

  • 僵尸状态是一个比较特殊的状态,当进程退出并且父进程没有读取到子进程退出的返回代码时,就会产生僵尸进程
  • 僵尸进程会以终止状态保持在进程表中,并且一致在等待父进程读取退出状态代码
  • 所以,子进程退出,但是父进程没有读取子进程状态,子进程加入Z状态
  • 系统调用exit,它的作用是使进程退出,但也仅仅限于将一个正常的进程变成一个僵尸进程,并不能将其完全销毁

一个僵尸进程的例子

一个进程使用fork创建子进程,如果子进程退出,而父进程并没有调用wait/waitpid获取子进程的状态信息,那么子进程的进程描述符仍然保存在系统中,这种进程就称之为僵尸进程

  1 #include<stdio.h>
  2 #include<errno.h>
  3 #include<stdlib.h>
  4 
  5 int main()
  6 {
  7     pid_t id = fork();
  8     if(id<0)
  9     {
 10         perror("fork");
 11         return 1;
 12     }
 13     else if(id>0){//father  
 14         printf("I an father\n");
 15         sleep(3);
 16         system("ps -o pid,ppid,state,tty,command");
 17         printf("father is exiting\n");
 18         }
 19     else{
 20             printf("I am child process ,I am exiting.\n");
 21             exit(0);
 22             }
 23     return 0;
 24 }

运行结果如下: 

 

我们可以看到,子进程27675使僵尸进程

僵尸进程的危害:

  • 进程的退出状态必须被维持下去,因为他要告诉父进程,你交给我的任务,我办的怎么样了,可父进程如果一直不读取,那么子进程就会一直处于Z状态
  • 维护退出状态本身就要用内存的数据维护,也属于进程基本信息,所以存在task_struct(PBC)中,换句话说,Z状态一直不退出,PCB就一直要维护它
  • 如果一个父进程创建了很多子进程,就是不回收,就会造成内存资源的浪费。因为数据结构对象本身就要占内存,如果数量过多就会造成内存泄漏

僵尸进程的避免:

  • 父进程可以通过wait和waitpid等待函数等待子进程结束,这会导致父进程挂起
  • 如果父进程很忙,那么可以用signal函数为SIGCHLD安装handler,因为子进程结束后,父进程会收到该信号,可以在handler中调用wait回收
  • 如果父进程不关心子进程什么时候结束,那么可以用signal(SIGCHLD,SIG_IGN)通知内核,自己对子进程的结束不感兴趣,那么子进程结束后,内核会回收,并不在给父进程发送信号
  • 还有一些技巧,就是fork俩次,父进程fork一个子进程,然后继续工作,子进程fork一个孙子进程后退出,那么孙子进程被init接管,孙子进程结束后,被init回收,不过子进程的回收还是要自己做

孤儿进程:

  • 父进程退出,子进程就被称之为“孤儿进程”
  • 孤儿进程被1号init进程领养,当然要有init进程回收

代码演示:

 1 #include<stdio.h>
  2 #include<unistd.h>
  3 #include<stdlib.h>
  4 #include<errno.h>
  5 
  6 int main()
  7 {
  8     pid_t id = fork();
  9     if(id<0)
 10     {
 11         perror("fork");
 12         return 1;
 13     }
 14     else if(id==0)
 15     {
 16         //child
 17         printf("I am child .");
 18         printf("pid: %d\tppid:%d\n",getpid(),getppid());
 19         printf("I will sleep ten ceconds.\n");
 20         sleep(10);
 21         printf("pid: %d\tppid:%d\n",getpid(),getppid());
 22         printf("child process id exited.\n");
 23     }
 24     else//father
 25     {
 26         printf("I am father,pid:%d\n",getpid());
 27         sleep(3);
 28         printf("father process is exited.\n");
 29         exit(0);
 30     }
 31     return 0;
 32 }

运行结果:

孤儿进程是没有父进程的进程,孤儿进程会被孤儿院收养(init进程)。每当出现一个孤儿进程的时候,内核就把孤儿进程的父进程设置为init,而init进程会循环的wait()它的已经退出的子进程。这样,当一个孤儿进程生命周期结束用户,init进程就会处理它的一切善后工作

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 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、付费专栏及课程。

余额充值