wait( )和 waitpid( )
進(jìn)程一旦調(diào)用了wait,就立即阻塞自己,由wait自動(dòng)分析是否當(dāng)前進(jìn)程的某個(gè)子進(jìn)程已經(jīng)退出,如果讓它找到了這樣一個(gè)已經(jīng)變成僵尸的子進(jìn)程,wait就會(huì)收集這個(gè)子進(jìn)程的信息,并把它徹底銷毀后返回;如果沒(méi)有找到這樣一個(gè)子進(jìn)程,wait就會(huì)一直阻塞在這里,直到有一個(gè)出現(xiàn)為止。
#include <sys/types.h> #include <sys/wait.h>pid_t wait(int *status)參數(shù)status用來(lái)保存被收集進(jìn)程退出時(shí)的一些狀態(tài),它是一個(gè)指向int類型的指針。但如果我們對(duì)這個(gè)子進(jìn)程是如何死掉的毫不在意,只想把這個(gè)僵尸進(jìn)程消滅掉,(事實(shí)上絕大多數(shù)情況下,我們都會(huì)這樣想),我們就可以設(shè)定這個(gè)參數(shù)為NULL
如果成功,wait會(huì)返回被收集的子進(jìn)程的進(jìn)程ID,如果調(diào)用進(jìn)程沒(méi)有子進(jìn)程,調(diào)用就會(huì)失敗,此時(shí)wait返回-1,同時(shí)errno被置為ECHILD。
下面是范例:
運(yùn)行后發(fā)現(xiàn),3秒鐘的等待時(shí)間,這就是我們?cè)O(shè)定的讓子進(jìn)程睡眠的時(shí)間,只有子進(jìn)程從睡眠中蘇醒過(guò)來(lái),它才能正常退出,也就才能被父進(jìn)程捕捉到。其實(shí)這里我們不管設(shè)定子進(jìn)程睡眠的時(shí)間有多長(zhǎng),父進(jìn)程都會(huì)一直等待下去。
數(shù)status:如果參數(shù)status的值不是NULL,wait就會(huì)把子進(jìn)程退出時(shí)的狀態(tài)取出并存入其中,這是一個(gè)整數(shù)值(int),指出了子進(jìn)程是正常退出還是被非正常結(jié)束的(一個(gè)進(jìn)程也可以被其他進(jìn)程用信號(hào)結(jié)束,我們將在以后的文章中介紹),以及正常結(jié)束時(shí)的返回值,或被哪一個(gè)信號(hào)結(jié)束的等信息。由于這些信息被存放在一個(gè)整數(shù)的不同二進(jìn)制位中,所以用常規(guī)的方法讀取會(huì)非常麻煩,人們就設(shè)計(jì)了一套專門的宏(macro)來(lái)完成這項(xiàng)工作,下面我們來(lái)學(xué)習(xí)一下其中最常用的兩個(gè):
1,WIFEXITED(status)這個(gè)宏用來(lái)指出子進(jìn)程是否為正常退出的,如果是,它會(huì)返回一個(gè)非零值。
(請(qǐng)注意,雖然名字一樣,這里的參數(shù)status并不同于wait唯一的參數(shù)–指向整數(shù)的指針status,而是那個(gè)指針?biāo)赶虻恼麛?shù),切記不要搞混了。)
2, WEXITSTATUS(status)當(dāng)WIFEXITED返回非零值時(shí),我們可以用這個(gè)宏來(lái)提取子進(jìn)程的返回值,如果子進(jìn)程調(diào)用exit(5)退出,WEXITSTATUS(status)就會(huì)返回5;如果子進(jìn)程調(diào)用exit(7),WEXITSTATUS(status)就會(huì)返回7。請(qǐng)注意,如果進(jìn)程不是正常退出的,也就是說(shuō),WIFEXITED返回0,這個(gè)值就毫無(wú)意義。
實(shí)例:
利用kill 殺死子進(jìn)程。
wait一次只能處理一個(gè)僵尸進(jìn)程:
// wait一次只能處理一個(gè)僵尸進(jìn)程 int main() {int count = 10;while (count--){pid_t pid = fork();switch (pid){case -1:perror ("fork");break;case 0: // 子進(jìn)程printf ("我是子進(jìn)程,我的ID是%d\n", getpid());sleep (3);exit (0);default: // 父進(jìn)程break;}}printf ("等待子進(jìn)程掛掉\n");while (1){pid_t childId = wait(NULL);if (-1 == childId){break;}printf ("成功處理一個(gè)子進(jìn)程,該子進(jìn)程是:%d\n", childId);}return 0; }#include <sys/types.h> #include <sys/wait.h>pid_t waitpid(pid_t pid,int *status,int options)
從本質(zhì)上講,系統(tǒng)調(diào)用waitpid和wait的作用是完全相同的,但waitpid多出了兩個(gè)可由用戶控制的參數(shù)pid和options,從而為我們編程提供了另一種更靈活的方式。下面我們就來(lái)詳細(xì)介紹一下這兩個(gè)參數(shù):
pid
從參數(shù)的名字pid和類型pid_t中就可以看出,這里需要的是一個(gè)進(jìn)程ID。但當(dāng)pid取不同的值時(shí),在這里有不同的意義。
pid>0時(shí),只等待進(jìn)程ID等于pid的子進(jìn)程,不管其它已經(jīng)有多少子進(jìn)程運(yùn)行結(jié)束退出了,只要指定的子進(jìn)程還沒(méi)有結(jié)束,waitpid就會(huì)一直等下去。
pid=-1時(shí),等待任何一個(gè)子進(jìn)程退出,沒(méi)有任何限制,此時(shí)waitpid和wait的作用一模一樣。
pid=0時(shí),等待同一個(gè)進(jìn)程組中的任何子進(jìn)程,如果子進(jìn)程已經(jīng)加入了別的進(jìn)程組,waitpid不會(huì)對(duì)它做任何理睬。
pid<-1時(shí),等待一個(gè)指定進(jìn)程組中的任何子進(jìn)程,這個(gè)進(jìn)程組的ID等于pid的絕對(duì)值。
options
options提供了一些額外的選項(xiàng)來(lái)控制waitpid,目前在Linux中只支持WNOHANG和WUNTRACED兩個(gè)選項(xiàng),這是兩個(gè)常數(shù),可以用”|”運(yùn)算符把它們連接起來(lái)使用
WNOHANG: 如果沒(méi)有任何已經(jīng)結(jié)束的子進(jìn)程則馬上返回,不予以等待。
WUNTRACED :如果子進(jìn)程進(jìn)入暫停執(zhí)行情況則馬上返回,但結(jié)束狀態(tài)不予以理會(huì)。
返回值:如果執(zhí)行成功則返回子進(jìn)程識(shí)別碼(PID),如果有錯(cuò)誤發(fā)生則返回-1。失敗原因存于errno中。
int main() {pid_t pid = fork();switch (pid){case -1:perror ("fork");break;case 0: // 子進(jìn)程printf ("我是子進(jìn)程,我的ID是%d\n", getpid());while (1);exit (0);default: // 父進(jìn)程printf ("等待子進(jìn)程掛掉\n");int status;pid_t childId = waitpid (-1, NULL, WNOHANG);printf ("成功處理一個(gè)子進(jìn)程,該子進(jìn)程是:%d, %d\n", childId, status);break;}return 0; }總結(jié)
以上是生活随笔為你收集整理的wait( )和 waitpid( )的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: anaconda python_机器学习
- 下一篇: java jpeg压缩解码_图片压缩(i