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