C语言僵尸进程,C/C++网络编程8——多进程服务器端之销毁僵尸进程
上一節(jié)提到,當(dāng)子進(jìn)程執(zhí)行結(jié)束,父進(jìn)程還在執(zhí)行,在父進(jìn)程結(jié)束之前子進(jìn)程會(huì)成為僵尸進(jìn)程,那么怎么銷毀僵尸進(jìn)程呢?父進(jìn)程主動(dòng)接收子進(jìn)程的返回值。
銷毀僵尸進(jìn)程的方法:
1:使用wait函數(shù)
2:使用waitpid函數(shù)
3:利用信號(hào)
1:使用wait函數(shù)銷毀僵尸進(jìn)程
#include pid_t wait(int *status);//成功返回終止的子進(jìn)程id,失敗返回-1
在父進(jìn)程中調(diào)用wait函數(shù)以后,如果有子進(jìn)程已經(jīng)執(zhí)行結(jié)束,那么子進(jìn)程傳回的返回值將存儲(chǔ)到status所指的內(nèi)存中,但是如果沒有子進(jìn)程執(zhí)行結(jié)束,父進(jìn)程將會(huì)阻塞在wait函數(shù),直到有子進(jìn)程執(zhí)行結(jié)束,這是一種不好的實(shí)現(xiàn)方法。
// wait函數(shù)銷毀僵尸進(jìn)程
#include #include#include
using namespacestd;intmain()
{
pid_t pid=fork();if (pid < 0) {
cout<< "fork() failed" <
}if (pid == 0) {return 5;
}else{int status = 0;
wait(&status);if (WIFEXITED(status)) { //子進(jìn)程正常結(jié)束
cout << "child return:" << WEXITSTATUS(status) <
}
sleep(300);
}return 0;
}
2:使用waitpid函數(shù)銷毀僵尸進(jìn)程
wait函數(shù)會(huì)使程序阻塞,可換用waitpid函數(shù)殺死僵尸進(jìn)程,同時(shí)能避免程序阻塞。
#include pid_t waitpid(pid_t pid,int *status, intoptions);//成功返回終止的子進(jìn)程id,失敗返回-1//pid : 等待終止的子進(jìn)程id,傳-1代碼任意子進(jìn)程終止//status:和wait函數(shù)的參數(shù)一樣//options:傳遞WNOHANG,沒有終止的子進(jìn)程也不進(jìn)入阻塞狀態(tài),返回0
#include #include#include
using namespacestd;intmain()
{
pid_t pid=fork();if (pid < 0) {
cout<< "fork() failed" <
}if (pid == 0) {
sleep(30);return 5;
}else{int status = 0;while (!waitpid(pid, &status, WNOHANG)) {
sleep(3);
cout<< "child proc is not finish" <
}if (WIFEXITED(status)) { //子進(jìn)程正常結(jié)束
cout << "child return:" << WEXITSTATUS(status) <
}
}return 0;
}
3:利用信號(hào)銷毀僵尸進(jìn)程
利用wait函數(shù)能銷毀僵尸進(jìn)程,但是會(huì)阻塞父進(jìn)程;利用waitpid也能銷毀僵尸進(jìn)程并且不阻塞父進(jìn)程,但是也需要不停的去檢查子進(jìn)程結(jié)束沒有。所以wait及waitpid方式都不完美。于是引入了信號(hào),信號(hào)是在特定事件發(fā)生時(shí)由操作系統(tǒng)向進(jìn)程發(fā)送的消息,接收到消息的進(jìn)程做信號(hào)處理。信號(hào)的使用方法是先注冊(cè)信號(hào),告訴操作系統(tǒng),當(dāng)某個(gè)信號(hào)發(fā)生時(shí),要做什么事情。
signal(SIGCHLD, do_what); SIGCHLD是子進(jìn)程結(jié)束的信號(hào),當(dāng)子進(jìn)程結(jié)束時(shí),執(zhí)行do_what函數(shù)。
其他常用信號(hào),SIGALRM:已到通過調(diào)用alarm函數(shù)注冊(cè)的時(shí)間;SIGINT:輸入CTRL+C
#include #include#include#include#include
using namespacestd;void read_child_proc(intsig)
{intstatus;
pid_t pid= waitpid(-1, &status, WNOHANG);if (WIFEXITED(status)) { //子進(jìn)程正常結(jié)束
cout << "child proc finish:" << pid <
}
}intmain()
{
pid_t pid=fork();if (pid < 0) {
cout<< "fork() failed" <
}
signal(SIGCHLD, read_child_proc);//注冊(cè)子進(jìn)程結(jié)束信號(hào),當(dāng)子進(jìn)程結(jié)束時(shí)調(diào)用read_child_proc函數(shù)
if (pid == 0) {return 5;
}else{
sleep(30);
cout<< "master proc wake up" <
}return 0;
}
執(zhí)行程序發(fā)現(xiàn)?master proc wake up并不是等30秒以后才輸出,而輸出很快,說明當(dāng)注冊(cè)子進(jìn)程結(jié)束信號(hào)以后,當(dāng)有子進(jìn)程執(zhí)行結(jié)束時(shí),會(huì)馬上喚醒sleep的進(jìn)程。
與50位技術(shù)專家面對(duì)面20年技術(shù)見證,附贈(zèng)技術(shù)全景圖總結(jié)
以上是生活随笔為你收集整理的C语言僵尸进程,C/C++网络编程8——多进程服务器端之销毁僵尸进程的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: c语言判断用户名重负,判定方阵的正定负定
- 下一篇: 西南交通大学计算机程序设计实验13,西南