孤兒進程: 因父親進程先退出而導致一個子進程被 init 進程收養的進程為孤兒進程,即孤兒進程的父親更改為 init 進程,該進程在孤兒進程退出後回收它的內核空間資源。 僵死進程: 進程已經退出,但它的父親進程還沒有回收內核資源的進程為僵死進程,即該進程在內核空間的 PCB(進程式控制制塊) 沒有釋放。 ...
孤兒進程:
因父親進程先退出而導致一個子進程被 init 進程收養的進程為孤兒進程,即孤兒進程的父親更改為 init 進程,該進程在孤兒進程退出後回收它的內核空間資源。
僵死進程:
進程已經退出,但它的父親進程還沒有回收內核資源的進程為僵死進程,即該進程在內核空間的 PCB(進程式控制制塊) 沒有釋放。
1.以下是一個孤兒進程的示常式序,在此程式中,讓父進程先退出,然後子進程再次輸出自己的父親進程號:
#include <stdio.h>
#include <unistd.h>
int main()
{
int pid;
if((pid = fork()) == -1)
perror("fork Err");
else if(!pid){
printf("Child: pid : %d, ppid : %d \n", getpid(), getppid());
sleep(2);
printf("Child: pid : %d, ppid : %d \n", getpid(), getppid());
}
else{
sleep(1);
printf("Parent: pid : %d, ppid : %d \n", getpid(), getppid());
}
return 0;
}
運行結果如下:
從結果來看,子進程的父進程前後發生了變化。
2.以下是僵死進程的示常式序,在此程式中,父進程讓子進程退出但不處理,然後父進程調用 system 函數列出當前前臺進程信息,代碼如下:
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
int main()
{
int pid;
if((pid = fork()) == -1)
perror("fork err");
else if(!pid){
exit(0);
}
sleep(1);
system("ps");
return 0;
}
運行結果如下:
紅框的進程即為僵死狀態。
怎樣來清除僵屍進程:
1.改寫父進程,在子進程死後要為它收屍。具體做法是接管SIGCHLD信號。子進程死後,會發送SIGCHLD信號給父進程,父進程收到此信號後,執行waitpid()函數為子進程收屍。
2.把父進程殺掉。父進程死後,僵屍進程成為"孤兒進程",過繼給1號進程init,init始終會負責清理僵屍進程.它產生的所有僵屍進程也跟著消失。
wait()與waitpid()
wait():
調用 wait() 函數的父親進程將阻塞式等待該進程的任意一個子進程結束後,回收該子進程的內核進程資源。
waitpid():
waitpid()函數可以用來等待指定子進程(指定PID的子進程)結束。函數聲明如下:
可以通過 /proc/{pid}/maps 文件查看進程資源。