信號集與信號屏蔽字 + APUE 第三版 程序10-15(信號設置與 sigprocmask 實例)

如有錯誤,歡迎批評指正,本人也是才學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 節)

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章