編程過程中,有時需要讓一個進程等待另一個進程,最常見的是父進程等待自己的子進程,或者父進程回收自己的子進程資源包括僵屍進程。這里簡單介紹一下系統調用函數:wait()
函數原型是
#include <sys/types.h>/* 提供類型pid_t的定義*/
#include <wait.h>
int wait(int *status)
函數功能是:父進程一旦調用了wait就立即阻塞自己,由wait自動分析是否當前進程的某個子進程已經退出,如果讓它找到了這樣一個已經變成僵屍的子進程,wait就會收集這個子進程的信息,並把它徹底銷毀后返回;如果沒有找到這樣一個子進程,wait就會一直阻塞在這里,直到有一個出現為止。
注:
當父進程忘了用wait()函數等待已終止的子進程時,子進程就會進入一種無父進程的狀態,此時子進程就是僵屍進程.
wait()要與fork()配套出現,如果在使用fork()之前調用wait(),wait()的返回值則為-1,正常情況下wait()的返回值為子進程的PID.
如果先終止父進程,子進程將繼續正常進行,只是它將由init進程(PID 1)繼承,當子進程終止時,init進程捕獲這個狀態.
參數status用來保存被收集進程退出時的一些狀態,它是一個指向int類型的指針。但如果我們對這個子進程是如何死掉毫不在意,只想把這個僵屍進程消滅掉,(事實上絕大多數情況下,我們都會這樣想),我們就可以設定這個參數為NULL,就像下面這樣:
pid = wait(NULL);
如果成功,wait會返回被收集的子進程的進程ID,如果調用進程沒有子進程,調用就會失敗,此時wait返回-1,同時errno被置為ECHILD。
如果參數status的值不是NULL,wait就會把子進程退出時的狀態取出並存入其中, 這是一個整數值(int),指出了子進程是正常退出還是被非正常結束的,以及正常結束時的返回值,或被哪一個信號結束的等信息。由於這些信息 被存放在一個整數的不同二進制位中,所以用常規的方法讀取會非常麻煩,人們就設計了一套專門的宏(macro)來完成這項工作,下面我們來學習一下其中最常用的兩個:
1,WIFEXITED(status) 這個宏用來指出子進程是否為正常退出的,如果是,它會返回一個非零值。
(請注意,雖然名字一樣,這里的參數status並不同於wait唯一的參數–指向整數的指針status,而是那個指針所指向的整數,切記不要搞混了。)
2, WEXITSTATUS(status) 當WIFEXITED返回非零值時,我們可以用這個宏來提取子進程的返回值,如果子進程調用exit(5)退出,WEXITSTATUS(status) 就會返回5;如果子進程調用exit(7),WEXITSTATUS(status)就會返回7。請注意,如果進程不是正常退出的,也就是說, WIFEXITED返回0,這個值就毫無意義。
代碼示例:wait.c
1 #include <stdio.h>
2 #include <sys/stat.h>
3 #include <sys/types.h>
4 #include <unistd.h>
5 #include <wait.h>
6 #include <errno.h>
7 #include <stdlib.h>
8 /***********************************************************
9 功能說明:進程等待wait()方法的應用
10 author: linux.sir@qq.com
11
12 ***********************************************************/
13 void waitprocess();
14
15
16 int main(int argc, char * argv[])
17 {
18 waitprocess();
19
20 }
21
22 void waitprocess()
23 {
24
25 int count = 0;
26
27 pid_t pid = fork();
28 int status = -1;
29
30 if(pid<0)
31 {
32 printf("fork error for %m\n",errno );
33 }else if(pid>0)
34 {
35 printf("this is parent ,pid = %d\n",getpid() );
36 wait(&status);//父進程執行到此,馬上阻塞自己,直到有子進程結束。當發現有子進程結束時,就會回收它的資源。
37
38 }else
39 {
40 printf("this is child , pid = %d , ppid = %d\n",getpid(),getppid() );
41 int i;
42
43 for (i = 0; i < 10; i++) {
44 count++;
45 sleep(1);
46
47 printf("count = %d\n", count) ;
48
49 }
50
51 exit(5);
52
53 }
54 printf("child exit status is %d\n", WEXITSTATUS(status));//status是按位存儲的狀態信息,需要調用相應的宏來還原一下
55
56 printf("end of program from pid = %d\n",getpid() );
57
58
59 }
運行結果:
原文地址:http://www.cnblogs.com/linux-sir/archive/2012/01/27/2330014.html