這是一個網友的提問:
1、若父進程已被waitpid阻塞,在子進程返回時,此時在父進程中SIGCHLD被阻塞(BLOCK),父進程收不到SIGCHLD信號,waitpid()函數能否正確返回,收集到子進程的信息?
2、 waitpid若能正確完成,在以后父進程中,將SIGCHLD信號UNBLOCK,用sigprocmask()函數解鎖,書上說,在 sigprocmask()函數返回以前,會將以前阻塞的信號發送給進程,父進程是否還能收到SIGCHLD信號?若能收到何必在開始時將SIGCHLD 進程阻塞。
簡單的對這個問題的解釋是wait及其變體并不是通過sigchld信號來知道子進程狀態的。
sigprocmask 阻塞的是有signal或sigaction設置的信號處理程序,即帶有SIGCHLD_Handle()等處理函數。wait不是靠接收sigchld 信號獲得子進程的退出狀態的,如果進程中同時設置了signal和wait,則子進程退出后發出sigchld信號,交到signal的信號處理程序處 理,wait接收到子進程退出狀態。
只是接收sigchld,而不調用wait還是會使子進程僵死的。一般的只有調用wait才能使子進程不成為僵死進程(除了2次fork 等或其他一些手段)。
只是接收sigchld,而不調用wait還是會使子進程僵死的。一般的只有調用wait才能使子進程不成為僵死進程(除了2次fork 等或其他一些手段)。
概括下:waitpid不是依靠SIGCHLD是否到達來判斷子進程是否退出,但是如果設置了SIGCHLD的處理函數,那么就需要等待SIGCHLD信號 的發生并完成信號處理函數,waitpid才能接收到子進程的退出狀態。在APUE中的system()實現中阻塞了SIGCHLD信號,但是并沒有設置 信號處理函數,所以waitpid在阻塞了SIGCHLD的情況下依然能正常返回,因為SIGCHLD在未設置信號處理函數的情況下不會影響到 waitpid的工作。至于為什么要阻塞SIGCHLD信號呢?那就是為了防止其他程序(main除了會調用system還會使用其他程序)設置了 SIGCHLD的信號處理函數,如果其他程序設置了SIGCHLD信號處理函數,在waitpid等待子程序的返回前,要去處理SIGCHLD信號處理程 序,如果阻塞了該信號,就不會去處理該信號處理程序,防止多余信息在system()中的出現。