Linux学习之系统编程篇:利用 SIGCHLD 回收子进程
生活随笔
收集整理的這篇文章主要介紹了
Linux学习之系统编程篇:利用 SIGCHLD 回收子进程
小編覺得挺不錯的,現在分享給大家,幫大家做個參考.
子進程死亡會給父進程發送 SIGCHLD 信號!(默認動作:父進程忽略該信號)此時父進程就可以捕捉該信號,利用回調函數使用 waitpid 回收子進程。
#include <stdio.h> #include <unistd.h> #include <signal.h> #include <sys/types.h> #include <sys/wait.h> void catch_sig(int sig) // 參數:是捕捉的信號編號 {pid_t pid; while(1) {pid = waitpid(-1, NULL, WNOHANG);if(pid <= 0) {break;}printf("died child_id=%d\n",pid);} } int main() {// 問題:父進程注冊捕捉的時候,子進程已經死亡,這樣信號根本捕捉不到// 重要思想:先屏蔽信號,再解除屏蔽// 這樣:即使子進程先死,內核產生的信號也會被屏蔽掉,然后再解除屏蔽,信號就可以被順利捕捉了// 父進程先設置屏蔽信號集,屏蔽SIGCHLD信號sigset_t proc; // 定義自定義信號集sigemptyset(&proc); // 第一步:清空自定義信號集,防止隨機數干擾其他信號sigaddset(&proc, SIGCHLD); // 將 sigchld 信號 添加到自定義集合sigprocmask(SIG_BLOCK, &proc, NULL); // 通過 自定義信號集 設置 屏蔽信號集// 創建5個子進程 int i = 0; for(i = 0 ;i < 5; i ++) {pid_t pid = fork();if(pid == 0){break;}}// 此時,6個進程在運行if(i < 5){// 子進程printf("I am %d child, mypid=%d\n", i+1, getpid());}else if(i == 5){// 父進程邏輯 前期已經將信號屏蔽掉了,此時使用 sigaction 進行捕捉struct sigaction act;act.sa_flags = 0;act.sa_handler = catch_sig;sigemptyset(&act.sa_mask); // 不設置臨時屏蔽信號sigaction(SIGCHLD, &act, NULL); // 注意:先注冊捕捉,再解除屏蔽,順序不能反sigprocmask(SIG_UNBLOCK, &proc, NULL);while(1){sleep(1); // 父進程一直存活,使子進程先執行return死亡}}return 0; } 與50位技術專家面對面20年技術見證,附贈技術全景圖總結
以上是生活随笔為你收集整理的Linux学习之系统编程篇:利用 SIGCHLD 回收子进程的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: TCP往返时延的估计和超时
- 下一篇: Linux学习之系统编程篇:守护进程(精