僵尸进程 wait() waitpid()

转载 2012年03月25日 16:52:17
       请先看前一篇关于exit函数于_exit函数的区别:http://blog.csdn.net/ll2323001/article/details/7392615
       如果我们已经了解了父进程和子进程的概念,并已经掌握了系统调用exit的用法,但可能很少有人意识到, 在一个进程调用了exit之后,该进程并非马上就消失掉,而是留下一个称为僵尸进程(Zombie)的数据结构。在Linux进程的5种状态中,僵尸进程 是非常特殊的一种,它已经放弃了几乎所有内存空间,没有任何可执行代码,也不能被调度,仅仅在进程列表中保留一个位置,记载该进程的退出状态等信息供其他 进程收集,除此之外,僵尸进程不再占有任何内存空间。从这点来看,僵尸进程虽然有一个很酷的名字,但它的影响力远远抵不上那些真正的僵尸兄弟,真正的僵尸 总能令人感到恐怖,而僵尸进程却除了留下一些供人凭吊的信息,对系统毫无作用。

也许读者们还对这个新概念比较好奇,那就让我们来看一眼Linux里的僵尸进程究竟长什么样子。

当一个进程已退出,但其父进程还没有调用系统调用wait(稍后介绍)对其进行收集之前的这段时间里,它会一直保持僵尸状态,利用这个特点,我们来写一个简单的小程序:

/* zombie.c */
#include<stdlib.h>
#include <unistd.h>
#include<sys/waid.h>
#include<sys/types.h>

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

   if(pid<0) /* 如果出错 */
      printf("error occurred!n");
   else if(pid==0) /* 如果是子进程 */
      exit(0);
   else  /* 如果是父进程 */
   {
      sleep(60); /* 休眠60秒,这段时间里,父进程什么也干不了 */
      wait(NULL); /* 收集僵尸进程 */
   }
   return 0;
}


sleep的作用是让进程休眠指定的秒数,在这60秒内,子进程已经退出,而父进程正忙着睡觉,不可能对它进行收集,这样,我们就能保持子进程60秒的僵尸状态。

编译这个程序:

$ cc zombie.c -o zombie


后台运行程序,以使我们能够执行下一条命令:

$ ./zombie &
[1] 1577


列一下系统内的进程:

$ ps -ax
...       ...
 1177 pts/0         S           0:00 -bash
 1577 pts/0         S           0:00 ./zombie
 1578 pts/0         Z           0:00 [zombie ]
 1579 pts/0         R           0:00 ps -ax

没有出现Z的zombie

看到中间的"Z"了吗?那就是僵尸进程的标志,它表示1578号进程现在就是一个僵尸进程。

我们已经学习了系统调用exit,它的作用是使进程退出,但也仅仅限于将一个正常的进程变成一个僵尸进程,并不能将其完全销毁。僵尸进程虽然对其他进程几 乎没有什么影响,不占用CPU时间,消耗的内存也几乎可以忽略不计,但有它在那里呆着,还是让人觉得心里很不舒服。而且Linux系统中进程数目是有限制 的,在一些特殊的情况下,如果存在太多的僵尸进程,也会影响到新进程的产生。那么,我们该如何来消灭这些僵尸进程呢?

先来了解一下僵尸进程的来由,我们知道,Linux和UNIX总有着剪不断理还乱的亲缘关系,僵尸进程的概念也是从UNIX上继承来的,而UNIX的先驱 们设计这个东西并非是因为闲来无聊想烦烦其他的程序员。僵尸进程中保存着很多对程序员和系统管理员非常重要的信息,首先,这个进程是怎么死亡的?是正常退 出呢,还是出现了错误,还是被其它进程强迫退出的?其次,这个进程占用的总系统CPU时间和总用户CPU时间分别是多少?发生页错误的数目和收到信号的数 目。这些信息都被存储在僵尸进程中,试想如果没有僵尸进程,进程一退出,所有与之相关的信息都立刻归于无形,而此时程序员或系统管理员需要用到,就只好干 瞪眼了。

那么,我们如何收集这些信息,并终结这些僵尸进程呢?就要靠我们下面要讲到的waitpid调用和wait调用。这两者的作用都是收集僵尸进程留下的信息,同时使这个进程彻底消失。下面就对这两个调用分别作详细介绍。

对于进程的一生可以用一些形象的比喻作一个小小的总结:
随着一句fork,一个新进程呱呱落地,但它这时只是老进程的一个克隆。
然后随着exec,新进程脱胎换骨,离家独立,开始了为人民服务的职业生涯。
人有生老病死,进程也一样,它可以是自然死亡,即运行到main函数的最后一个”}”,从容地离我们而去;也可以是自杀,自杀有2种方式,一种是调用 exit函数,一种是在main函数内使用return,无论哪一种方式,它都可以留下遗书,放在返回值里保留下来;它还甚至能可被谋杀,被其它进程通过 另外一些方式结束他的生命。
进程死掉以后,会留下一具僵尸,wait和waitpid充当了殓尸工,把僵尸推去火化,使其最终归于无形。

linux中wait系统调用一文中介绍了其中的一个殓尸工wait, 下面介绍另一个waitpid,这个貌似复杂些。

waitpid函数原型:

#include<sys/types.h>/* 提供类型pid_t的定义 */
#include<sys/wait.h>
pid_t waitpid(pid_tpid,int* status,intoptions);

从本质上讲,系统调用waitpid和wait的作用是完全相同的,但waitpid多出了两个可由用户控制的参数pid和options,从而为我们编程提供了另一种更灵活的方式。下面我们就来详细介绍一下这两个参数:

pid

从参数的名字pid和类型pid_t中就可以看出,这里需要的是一个进程ID。但当pid取不同的值时,在这里有不同的意义。

1. pid>0时,只等待进程ID等于pid的子进程,不管其它已经有多少子进程运行结束退出了,只要指定的子进程还没有结束,waitpid就会一直等下去。
2. pid=-1时,等待任何一个子进程退出,没有任何限制,此时waitpid和wait的作用一模一样。
3. pid=0时,等待同一个进程组中的任何子进程,如果子进程已经加入了别的进程组,waitpid不会对它做任何理睬。
4. pid<-1时,等待一个指定进程组中的任何子进程,这个进程组的ID等于pid的绝对值。

options

options提供了一些额外的选项来控制waitpid,目前在Linux中只支持WNOHANG和WUNTRACED两个选项,这是两个常数,可以用”|”运算符把它们连接起来使用,比如:

ret=waitpid(-1,NULL,WNOHANG|WUNTRACED);

如果我们不想使用它们,也可以把options设为0,如:

ret=waitpid(-1,NULL,0);

如果使用了WNOHANG参数调用waitpid,即使没有子进程退出,它也会立即返回,不会像wait那样永远等下去。

而WUNTRACED参数,用于跟踪调试,极少用到,就不说了。

查看linux源代码 unistd.h 我们会发现,其实 wait 就是经过包装的 waitpid:

static inline pid_t wait(int*wait_stat)
{
      
return waitpid(-1,wait_stat,0);
}

waitpid的返回值比wait稍微复杂一些,一共有3种情况:

1. 当正常返回的时候,waitpid返回收集到的子进程的进程ID;
2. 如果设置了选项WNOHANG,而调用中waitpid发现没有已退出的子进程可收集,则返回0;
3. 如果调用中出错,则返回-1,这时errno会被设置成相应的值以指示错误所在;

当pid所指示的子进程不存在,或此进程存在,但不是调用进程的子进程,waitpid就会出错返回,这时errno被设置为ECHILD;

下面看一个简单的例子:

下载:waitpid.c
  1. /* waitpid.c */
  2. #include<sys/types.h>
  3. #include<sys/wait.h>
  4. #include<unistd.h>
  5. #include<stdio.h>

  6. int  main(void)
  7. {
  8.       pid_tpc,pr;

  9.       pc=fork();
  10.       if(pc<0)/* fork错误*/
  11.       {
  12.            printf("fork error\n");
  13.            exit(1);
  14.       }
  15.      elseif(pc==0)/*在子进程中*/
  16.      {
  17.            sleep(10);
  18.            exit(0);
  19.      }
  20.      else                 //  注意:fork返回值大于0时,返回的是子进程的ID
  21.     {
  22.            do
  23.           {/* 使用了WNOHANG参数,waitpid不会在这里等待 */
  24.                   pr=waitpid(pc,NULL,WNOHANG);
  25.                   if(pr==0)
  26.                   {
  27.                        printf("No child exit\n");
  28.                        sleep(1);
  29.                   }
  30.            }while(pr==0);
  31.            if(pr==pc)
  32.                    printf("successfully get child %d\n",pr);
  33.            else
  34.                    printf("wait child error\n");
  35.       }
  36.       return  0;
  37. }

编译并运行:

$ gcc -o waitpid waitpid.c
$ ./waitpid
No child exit
No child exit
No child exit
No child exit
No child exit
No child exit
No child exit
No child exit
No child exit
No child exit
successfully get child 4607

父进程经过10次失败的尝试之后,终于收集到了退出的子进程。父进程和子进程分别睡眠了10秒钟和1秒钟,代表它们分别作了10秒钟和1秒钟的工作。父子进程都有工作要做,父进程利用工作的简短间歇察看子进程的是否退出,如退出就收集它。


僵尸进程以及wait和waitpid函数

基本概念:  我们知道在unix/linux中,正常情况下,子进程是通过父进程创建的,子进程再创建新的进程。子进程的结束和父进程的运行是一个异步过程,即父进程永远无法预测子进程 到底什么时候结束。 当...
  • fly__chen
  • fly__chen
  • 2016年11月21日 14:18
  • 772

waitpid和SIGCHLD信号回收僵尸进程

对于多进程而言,父进程一般需要跟踪子进程的退出状态。因此当子进程结束运行时,内核不会立即释放该进程的进程表的表项。以满足父进程后续对子进程退出的信息查询(死后验尸),当然前提是父进程还在运行。在子进程...
  • huangshanchun
  • huangshanchun
  • 2015年08月09日 08:28
  • 1985

waitpid函数和wait函数的用法和区别

 #include #include pid_t wait(int *status) 进程一旦调用了wait,就立即阻塞自己,由wait自动分析是否当前进程的某个子进程已经退出,如果...
  • xilihong816
  • xilihong816
  • 2016年10月23日 10:18
  • 1922

僵尸进程 wait waitpid

请先看前一篇关于exit函数于_exit函数的区别:http://blog.csdn.net/ll2323001/article/details/7392615        如果我们已经了解了父进...
  • qq_32625545
  • qq_32625545
  • 2016年01月22日 20:47
  • 122

僵尸进程 wait() waitpid()

如果我们已经了解了父进程和子进程的概念,并已经掌握了系统调用exit的用法,但可能很少有人意识到, 在一个进程调用了exit之后,该进程并非马上就消失掉,而是留下一个称为僵尸进程(Zombie)的数据...
  • wocjj
  • wocjj
  • 2012年04月03日 14:26
  • 690

wait、waitpid及僵尸进程

一wait 父进程等待子进程结束 #include /* 提供类型pid_t的定义 */ #include pid_t wait(int *status) 进程一旦调用了wait,就立...
  • chm880910
  • chm880910
  • 2015年10月18日 21:20
  • 218

Linux wait/waitpid详解

Linux wait/waitpid详解
  • zhangxiao93
  • zhangxiao93
  • 2017年06月04日 15:39
  • 553

wait waitpid WNOHANG 僵尸进程

什么是僵尸进程? 首先内核会释放终止进程(调用了exit系统调用)所使用的所有存储区,关闭所有打开的文件等,但内核为每一个终止子进程保存了一定量的信息。这些信息至少包括进程ID,进程的终止状态,以及该...
  • eastlhu
  • eastlhu
  • 2014年03月05日 21:56
  • 824

linux c之wait和waitpid函数的用法和总结

1、wait和waitpid函数的介绍        1)  wait()函数用于使父进程(也就是调用wait()的进程)阻塞,直到一个子进程结束或者该进程接收到了一个指定的信号为止。如果该父进...
  • u011068702
  • u011068702
  • 2017年01月13日 17:56
  • 3722

UNIX网络编程——使用waitpid处理僵尸进程(TCP客户/服务器优化1)

1、僵尸进程 僵尸进程:一个进程使用fork创建子进程,如果子进程退出,而父进程并没有调用wait或waitpid获取子进程的状态信息,那么子进程的进程描述符仍然保存在系统中。这种进程称之为...
  • songshiMVP1
  • songshiMVP1
  • 2016年07月06日 12:01
  • 449
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:僵尸进程 wait() waitpid()
举报原因:
原因补充:

(最多只允许输入30个字)