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,就象下面这样ide

pid = wait(NULL);  函数

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

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

 1     #include <sys/types.h>  
 2       
 3     #include <sys/wait.h>  
 4       
 5     #include <unistd.h>  
 6       
 7     #include <stdlib.h>  
 8       
 9     main()  
10       
11     {  
12       
13          pid_t pc,pr;  
14       
15             pc=fork();  
16       
17       if(pc<0)                   
18       
19                 printf("error ocurred!/n");  
20       
21      else if(pc==0){            
22       
23                printf("This is child process with pid of %d/n",getpid());  
24       
25               sleep(10);        
26       
27         }  
28       
29        else{                     
30       
31                 pr=wait(NULL);    
32       
33                 printf("I catched a child process with pid of %d/n"),pr);  
34       
35        }                 
36       
37         exit(0);  
38       
39     } 

 编译并运行:指针

1     $ cc wait1.c -o wait1  
2       
3     $ ./wait1  
4       
5     This is child process with pid of 1508  
6       
7     I catched a child process with pid of 1508 
运行结果

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

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

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

下面经过例子来实战一下咱们刚刚学到的内容:blog

 1     #include <sys/types.h>  
 2       
 3     #include <sys/wait.h>  
 4       
 5     #include <unistd.h>  
 6       
 7     main()  
 8       
 9     {  
10       
11         int status;  
12       
13      pid_t pc,pr;  
14       
15             pc=fork();  
16       
17       if(pc<0)   
18       
19                 printf("error ocurred!/n");  
20       
21      else if(pc==0){   
22       
23                 printf("This is child process with pid of %d./n",getpid());  
24       
25              exit(3);          
26       
27         }  
28       
29        else{             
30       
31                 pr=wait(&status);  
32       
33                if(WIFEXITED(status)){    
34       
35                         printf("the child process %d exit normally./n",pr);  
36       
37                      printf("the return code is %d./n",WEXITSTATUS(status));  
38       
39                  }else                     
40       
41                         printf("the child process %d exit abnormally./n",pr);  
42       
43            }  
44       
45     } 

 编译并运行:

1         $ cc wait2.c -o wait2  
2           
3         $ ./wait2  
4           
5         This is child process with pid of 1538.  
6           
7         the child process 1538 exit normally.  
8           
9         the return code is 3.
运行结果

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

进程同步:

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

 1     #include <sys/types.h>  
 2       
 3     #include <sys/wait.h>  
 4       
 5     main()  
 6       
 7     {  
 8       
 9        pid_t pc, pr;  
10       
11            int status;  
12       
13        
14       
15         pc=fork();  
16       
17         
18       
19         if(pc<0)  
20       
21                  printf("Error occured on forking./n");  
22       
23           else if(pc==0){  
24       
25                    
26       
27                 exit(0);  
28       
29         }else{  
30       
31                     
32       
33                 pr=wait(&status);  
34       
35                  
36       
37         }  
38       
39     } 

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

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

1     #include <sys/types.h>   
2       
3     #include <sys/wait.h>  
4       
5     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     {  
4       
5       return waitpid(-1,wait_stat,0);  
6       
7     } 

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

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

 1     #include <sys/types.h>  
 2       
 3     #include <sys/wait.h>  
 4       
 5     #include <unistd.h>  
 6       
 7     main()  
 8       
 9     {  
10       
11           pid_t pc, pr;  
12       
13                      
14       
15         pc=fork();  
16       
17       if(pc<0)           
18       
19                 printf("Error occured on forking./n");  
20       
21           else if(pc==0){           
22       
23                 sleep(10);        
24       
25                 exit(0);  
26       
27         }  
28       
29          
30       
31         do{  
32       
33              pr=waitpid(pc, NULL, WNOHANG);    
34       
35                 if(pr==0){                        
36       
37                         printf("No child exited/n");  
38       
39                             sleep(1);  
40       
41                }  
42       
43        }while(pr==0);                            
44       
45         if(pr==pc)  
46       
47               printf("successfully get child %d/n", pr);  
48       
49       else  
50       
51                     printf("some error occured/n");  
52       
53     }  

编译并运行:

 1     $ cc waitpid.c -o waitpid  
 2       
 3     $ ./waitpid  
 4       
 5     No child exited  
 6       
 7     No child exited  
 8       
 9     No child exited  
10       
11     No child exited  
12       
13     No child exited  
14       
15     No child exited  
16       
17     No child exited  
18       
19     No child exited  
20       
21     No child exited  
22       
23     No child exited  
24       
25     successfully get child 1526 
运行结果

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

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

相关文章
相关标签/搜索