如有錯誤,歡迎批評指正,本人也是才學APUE的菜鳥
信號集:可以理解爲是一個整型,其每一個二進制位都代表一種信號。1 表示在這個信號集中,0 表示不在這個信號集中。唯一需要注意的一點:由於不存在編號爲 0 的信號,所以信號編號爲 i 的信號對應的其實是信號集中的第 i - 1 的二進制位。
所以其四個操作函數:
- sigemptyset(sigset_t *set)
- sigfillset(sigset_t *set)
- sigaddset(sigset_t set, int signo)
- sigdelset(sigset_t set, int signo)
都很容易通過位運算來實現。其中,函數的作用都是函數名自解釋的。
信號屏蔽字:信號屏蔽字是一個信號集,它規定了當前阻塞而不能遞送給該進程的信號。(注意:SIGKILL 和 SIGSTOP 是不能阻塞的)
這裏有一個函數 sigprocmask 用於檢測或更改信號屏蔽字。(注意:sigprocmask 是僅爲單線程定義的)
- sigprocmask(int how, const sigset_t *restrict set, sigset_t *oset);
這裏還需要提一個函數:sigpending,它返回進程的未決信號集。
10-15 程序 Code:
#include "apue.h"
static void sig_quit(int);
int main(void) {
sigset_t newmask, oldmask, pendmask;
if(signal(SIGQUIT, sig_quit) == SIG_ERR) // 捕捉 SIGQUIT 信號
err_sys("can't catch SIGQUIT");
/*
* Block SIGQUIT and save current signal mask.
*/
sigemptyset(&newmask); // empty 初始化信號集 newmask
sigaddset(&newmask, SIGQUIT); // 向 newmask 中添加 SIGQUIT 信號
// 將當前進程的信號屏蔽字改爲與 newmask 的並集,並將之前的信號屏蔽字放入 oldmask
if(sigprocmask(SIG_BLOCK, &newmask, &oldmask) < 0)
err_sys("SIG_BLOCK error");
sleep(5); // SIGQUIT here will remain pending。休眠 5 秒等待 SIGQUIT 的到來,但此信號會被阻塞,並加入進程未決信號集
if(sigpending(&pendmask) < 0) // 取進程未決的信號集,放入 pendmask
err_sys("sigpending error");
if(sigismember(&pendmask, SIGQUIT)) // 判斷 pendmask 是否含有 SIGQUIT 信號
printf("\nSIGQUIT pending\n");
/*
* Restore signal mask which unblocks SIGQUIT.
*/
if(sigprocmask(SIG_SETMASK, &oldmask, NULL) < 0) // 將信號屏蔽字復原爲 oldmask
err_sys("SIG_SETMASK error");
printf("SIGQUIT unblocked\n");
sleep(5); // SIGQUIT here will terminate with core file
exit(0);
}
static void sig_quit(int signo) {
printf("caught SIGQUIT\n");
if(signal(SIGQUIT, SIG_DFL) == SIG_ERR)
err_sys("can't reset SIGQUIT");
}
終端運行情況:
hjm@hjm-Inspiron:~/InterviewPreparation/apue/Examples/10-15$ ./10-15
^\
SIGQUIT pending
caught SIGQUIT
SIGQUIT unblocked
hjm@hjm-Inspiron:~/InterviewPreparation/apue/Examples/10-15$ ./10-15
^\^\^\^\^\^\^\^\^\^\^\^\
SIGQUIT pending
caught SIGQUIT
SIGQUIT unblocked
^\退出 (核心已轉儲)
^\ 是鍵入 ctrl + \,發出退出信號 SIGQUIT
輸出就不用說了,代碼註釋中解釋的很清楚。
對於第二次調用,雖然多次鍵入了退出字符,但是可以發現,caught 只輸出了一次。說明此係統上沒有將信號進行排隊(見 10.8 節)。