C函数篇(wait函数)

  1. #include <sys/types.h>
  2. #include <sys/wait.h>
  3. pid_t wait(int *status)

进程一旦调用了wait,就立即阻塞自己,由wait自动分析是否当前进程的某个子进程已经 退出,如果让它找到了这样一个已经变成僵尸的子进程,wait就会收集这个子进程的信息,并把它彻底销毁后返回;如果没有找到这样一个子进程,wait就 会一直阻塞在这里,直到有一个出现为止。

参数status用来保存被收集进程退出时的一些状态,它是一个指向int类型的指针。但如果我们对这个子进程是如何死掉的毫不在意,只想把这个僵尸进程消灭掉,(事实上绝大多数情况下,我们都会这样想),我们就可以设定这个参数为NULL,就象下面这样

pid = wait(NULL);

如果成功,wait会返回被收集的子进程的进程ID,如果调用进程没有子进程,调用就会失败,此时wait返回-1,同时errno被置为ECHILD。

下面就让我们用一个例子来实战应用一下wait调用:

  1. #include <sys/types.h>
  2. #include <sys/wait.h>
  3. #include <unistd.h>
  4. #include <stdlib.h>
  5. main()
  6. {
  7. pid_t pc,pr;
  8. pc=fork();
  9. if(pc<0)
  10. printf("error ocurred!/n");
  11. else if(pc==0){
  12. printf("This is child process with pid of %d/n",getpid());
  13. sleep(10);
  14. }
  15. else{
  16. pr=wait(NULL);
  17. printf("I catched a child process with pid of %d/n"),pr);
  18. }
  19. exit(0);
  20. }

编译并运行:

  1. $ cc wait1.c -o wait1
  2. $ ./wait1
  3. This is child process with pid of 1508
  4. I catched a child process with pid of 1508

可以明显注意到,在第2行结果打印出来前有10秒钟的等待时间,这就是我们设定的让子进程睡 眠的时间,只有子进程从睡眠中苏醒过来,它才能正常退出,也就才能被父进程捕捉到。其实这里我们不管设定子进程睡眠的时间有多长,父进程都会一直等待下 去,读者如果有兴趣的话,可以试着自己修改一下这个数值,看看会出现怎样的结果。数status:

1,WIFEXITED(status)这个宏用来指出子进程是否为正常退出的,如果是,它会返回一个非零值。

2, WEXITSTATUS(status)当WIFEXITED返回非零值时,我们可以用这个宏来提取子进程的返回值,如果子进程调用exit(5)退 出,WEXITSTATUS(status)就会返回5;如果子进程调用exit(7),WEXITSTATUS(status)就会返回7。请注意,如 果进程不是正常退出的,也就是说,WIFEXITED返回0,这个值就毫无意义。

下面通过例子来实战一下我们刚刚学到的内容:

  1. #include <sys/types.h>
  2. #include <sys/wait.h>
  3. #include <unistd.h>
  4. main()
  5. {
  6. int status;
  7. pid_t pc,pr;
  8. pc=fork();
  9. if(pc<0)
  10. printf("error ocurred!/n");
  11. else if(pc==0){
  12. printf("This is child process with pid of %d./n",getpid());
  13. exit(3);
  14. }
  15. else{
  16. pr=wait(&status);
  17. if(WIFEXITED(status)){
  18. printf("the child process %d exit normally./n",pr);
  19. printf("the return code is %d./n",WEXITSTATUS(status));
  20. }else
  21. printf("the child process %d exit abnormally./n",pr);
  22. }
  23. }

编译并运行:

    1. $ cc wait2.c -o wait2
    2. $ ./wait2
    3. This is child process with pid of 1538.
    4. the child process 1538 exit normally.
    5. the return code is 3.

当然,处理进程退出状态的宏并不止这两个,但它们当中的绝大部分在平时的编程中很少用到,就也不在这里浪费篇幅介绍了,有兴趣的读者可以自己参阅Linuxman pages去了解它们的用法。

进程同步:

有时候,父进程要求子进程的运算结果进行下一步的运算,或者子进程的功能是为父进程提供了下 一步执行的先决条件(如:子进程建立文件,而父进程写入数据),此时父进程就必须在某一个位置停下来,等待子进程运行结束,而如果父进程不等待而直接执行 下去的话,可以想见,会出现极大的混乱。这种情况称为进程之间的同步,更准确地说,这是进程同步的一种特例。进程同步就是要协调好2个以上的进程,使之以 安排好地次序依次执行。解决进程同步问题有更通用的方法,我们将在以后介绍,但对于我们假设的这种情况,则完全可以用wait系统调用简单的予以解决。请 看下面这段程序:

  1. #include <sys/types.h>
  2. #include <sys/wait.h>
  3. main()
  4. {
  5. pid_t pc, pr;
  6. int status;
  7. pc=fork();
  8. if(pc<0)
  9. printf("Error occured on forking./n");
  10. else if(pc==0){
  11. exit(0);
  12. }else{
  13. pr=wait(&status);
  14. }
  15. }

这段程序只是个例子,不能真正拿来执行,但它却说明了一些问题,首先,当fork调用成功 后,父子进程各做各的事情,但当父进程的工作告一段落,需要用到子进程的结果时,它就停下来调用wait,一直等到子进程运行结束,然后利用子进程的结果 继续执行,这样就圆满地解决了我们提出的进程同步问题。

waitpid系统调用在Linux函数库中的原型是:

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

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

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

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

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

而WUNTRACED参数,由于涉及到一些跟踪调试方面的知识,加之极少用到,这里就不多费笔墨了,有兴趣的读者可以自行查阅相关材料。

看到这里,聪明的读者可能已经看出端倪了--wait不就是经过包装的waitpid吗?没错,察看<内核源码目录>/include/unistd.h文件349-352行就会发现以下程序段:

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

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

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

  1. #include <sys/types.h>
  2. #include <sys/wait.h>
  3. #include <unistd.h>
  4. main()
  5. {
  6. pid_t pc, pr;
  7. pc=fork();
  8. if(pc<0)
  9. printf("Error occured on forking./n");
  10. else if(pc==0){
  11. sleep(10);
  12. exit(0);
  13. }
  14. do{
  15. pr=waitpid(pc, NULL, WNOHANG);
  16. if(pr==0){
  17. printf("No child exited/n");
  18. sleep(1);
  19. }
  20. }while(pr==0);
  21. if(pr==pc)
  22. printf("successfully get child %d/n", pr);
  23. else
  24. printf("some error occured/n");
  25. }

编译并运行:

  1. $ cc waitpid.c -o waitpid
  2. $ ./waitpid
  3. No child exited
  4. No child exited
  5. No child exited
  6. No child exited
  7. No child exited
  8. No child exited
  9. No child exited
  10. No child exited
  11. No child exited
  12. No child exited
  13. successfully get child 1526

父进程经过10次失败的尝试之后,终于收集到了退出的子进程。

因为这只是一个例子程序,不便写得太复杂,所以我们就让父进程和子进程分别睡眠了10秒钟和1秒钟,代表它们分别作了10秒钟和1秒钟的工作。父子进程都有工作要做,父进程利用工作的简短间歇察看子进程的是否退出,如退出就收集它。

时间: 2024-07-28 13:41:06

C函数篇(wait函数)的相关文章

函数篇之——函数也是数据

1.  对于JavaScript 来说,有一个概念十分重要——既函数是一种数据类型. 也就是说,下面两种函数定义在本质上是相同的: ① . function f () { return 1; } ② . var f = function () { return 1; } 其中,第二种定义方式通常被叫做函数标识记法(function literal notation). 如果我们对函数变量调用typeof,操作符返回的字符串将会是“function”. >>> var f = functi

C函数篇(Timer函数)

语法 Timer() 语法Timer ( interval {, windowname } ) 参数 指定两次触发Timer事件之间的时间间隔,有效值在0到65之间.如果该参数的值指定为0,那么关闭定时器, 不再触发指定窗口的Timer事件.windowname:窗口名,指定时间间隔到时要触发哪个窗口的Timer事件.省略该参数时,触发当前窗口的 Timer事件返回值Integer.函数执行成功时返回1,发生错误时返回-1.如果任何参数的值为NULL,Timer()函数返回NULL.用法使 用T

C函数篇(setsockopt函数②)

套接口选项 在前面的几章中,我们讨论了使用套接口的基础内容.现在我们要来探讨一些可用的其他的特征.在我们掌握了这一章的概念之后,我们就为后面的套接口的高级主题做好了准备.在这一章,我们将会专注于下列主题: 如何使用getsockopt(2)函数获得套接口选项值 如何使用setsockopt(2)函数设置套接口选项值 如何使用这些常用的套接口选项 得到套接口选项 有时,一个程序需要确定为当前为一个套接口进行哪些选项设置.这对于一个子程序库函数尤其如此,因为这个库函数并不知道为这个套接口进行哪些设置

SQl Server 函数篇 数学函数,字符串函数,转换函数,时间日期函数

数据库中的函数和c#中的函数很相似 按顺序来, 这里价格特别的 print  可以再消息栏里打印东西 数学函数 ceiling()  取上限   不在乎小数点后面有多大,直接忽略 floor()     取下限   同上 round(列名,保留的位数)   四舍五入   保留小数最后那位数进不进一只看保留位数的后一位数够不够条件,再往后的就不管了 ABS()     绝对值---防抱死233 PI()        圆周率   就是查询一个圆周率 SQRT()平方根 字符串函数 upper()

C函数篇(socket函数)

简述 创建一个套接口. #include <winsock.h> SOCKET PASCAL FAR socket( int af, int type, int protocol); af:一个地址描述.目前仅支持AF_INET格式,也就是说ARPA Internet地址格式. type:新套接口的类型描述. protocol:套接口所用的协议.如调用者不想指定,可用0指定,表示缺省. 注释       socket()函数用于根据指定的地址族.数据类型和协议来分配一个套接口的描述字及其所用的

C函数篇(select函数)

Select在Socket编程中还是比较重要的,可是对于初学Socket的人来说都不太爱用Select写程序,他们只是习惯写诸如 connect.accept.recv或recvfrom这样的阻塞程序(所谓阻塞方式block,顾名思义,就是进程或是线程执行到这些函数时必须等 待某个事件的发生,如果事件没有发生,进程或线程就被阻塞,函数不能立即返回).可是使用Select就可以完成非阻塞(所谓非阻塞方式non- block,就是进程或线程执行此函数时不必非要等待事件的发生,一旦执行肯定返回,以返回

C函数篇(fork函数)

一.fork入门知识 一个进程,包括代码.数据和分配给进程的资源.fork()函数通过系统调用创建一个与原来进程几乎完全相同的进程,也就是两个进程可以做完全相同的事,但如果初始参数或者传入的变量不同,两个进程也可以做不同的事.    一个进程调用fork()函数后,系统先给新的进程分配资源,例如存储数据和代码的空间.然后把原来的进程的所有值都复制到新的新进程中,只有少数值与原来的进程的值不同.相当于克隆了一个自己. 我们来看一个例子: /* *  fork_test.c *  version 1

C函数篇(send函数)

send()用于向一个已经连接的socket发送数据,如果无错误,返回值为所发送数据的总数,否则返回SOCKET_ERROR. 简述 向一个已连接的套接口发送数据. #include <winsock.h> int PASCAL FAR send( SOCKET s, const char FAR* buf, int len, int flags); s:一个用于标识已连接套接口的描述字. buf:包含待发送数据的缓冲区. len:缓冲区中数据的长度. flags:调用执行方式. 注释 sen

C函数篇(recv函数)

简述 从一个套接口接收数据. 表头文件: #include<sys/types.h> #include<sys/socket.h> int PASCAL FAR recv( SOCKET s, char FAR* buf, int len, int flags); s:一个标识已连接套接口的描述字. buf:用于接收数据的缓冲区. len:缓冲区长度. flags:指定调用方式. 流程 这里只描述同步Socket的recv函数的执行流程.当应用程序调用recv函数时: (1)rec