在 CentOS 系統中,僵尸進程的出現通常是由于父進程沒有正確處理子進程的結束狀態。為了避免僵尸進程,可以采取以下幾種方法:
父進程應當在子進程結束后調用 wait()
或 waitpid()
函數來回收子進程的資源。這樣可以確保子進程的信息被正確清理,避免形成僵尸進程。
#include <sys/wait.h>
int main() {
pid_t pid;
if ((pid = fork()) == 0) { /* 子進程 */
exit(0);
} else { /* 父進程 */
int status;
wait(&status); // 等待子進程結束并回收資源
printf("子進程已結束, PID: %d
", pid);
}
return 0;
}
父進程可以設置信號處理函數來捕獲子進程結束的信號(SIGCHLD),在信號處理函數中調用 wait()
或 waitpid()
來清理子進程資源。
#include <signal.h>
#include <sys/wait.h>
#include <unistd.h>
#include <stdio.h>
void sigchld_handler(int signo) {
while (waitpid(-1, NULL, WNOHANG) > 0) {
// 處理已結束的子進程
}
}
int main() {
struct sigaction sa;
sa.sa_handler = sigchld_handler;
sigemptyset(&sa.sa_mask);
sa.sa_flags = SA_RESTART | SA_NOCLDSTOP;
sigaction(SIGCHLD, &sa, NULL);
pid_t pid = fork();
if (pid == 0) { /* 子進程 */
exit(0);
} else { /* 父進程 */
// 父進程可以繼續執行其他任務
sleep(5); // 模擬父進程執行其他任務
}
return 0;
}
signal(SIGCHLD, SIG_IGN)
忽略 SIGCHLD 信號如果父進程不關心子進程的結束狀態,可以設置信號處理函數為忽略 SIGCHLD 信號。這樣,子進程結束后,內核會自動回收其資源,不會形成僵尸進程。
#include <signal.h>
#include <unistd.h>
#include <stdio.h>
int main() {
signal(SIGCHLD, SIG_IGN); // 忽略 SIGCHLD 信號
// 父進程可以繼續執行其他任務
return 0;
}
在父進程中再次調用 fork()
創建一個子進程,這樣第一個子進程會成為第二個子進程的父進程。當第一個子進程結束時,其資源會被第二個子進程回收,從而避免僵尸進程。
#include <sys/wait.h>
#include <unistd.h>
#include <stdio.h>
int main() {
pid_t pid = fork();
if (pid == 0) { /* 第一個子進程 */
pid_t pid2 = fork();
if (pid2 == 0) { /* 第二個子進程 */
exit(0); // 第二個子進程結束后,其資源會被init進程回收
}
exit(0); // 第一個子進程結束后,其資源會被init進程回收
} else if (pid > 0) { /* 父進程 */
// 父進程可以繼續執行其他任務
wait(NULL); // 等待第一個子進程結束并回收資源
} else {
perror("fork error");
exit(EXIT_FAILURE);
}
return 0;
}
可以使用 ps
命令結合 grep
來定期檢查系統中的僵尸進程,并手動或自動清理這些進程。
ps aux | grep 'Z'
通過以上方法,可以有效預防 CentOS 系統中出現僵尸進程,確保系統的穩定性和資源的合理利用。