先來回顧一下未決信號 集是怎麼回事。
信號從產生到抵達目的地,叫作信號遞達。而信號從產生到遞達的中間狀態,叫作信號的未決狀態。產生未決狀態的原因有可能是信號受到阻塞了,也就是信號屏蔽字(或稱阻塞信號集,mask)對應位被置1。阻塞信號集和未決信號集均是由內核維護的,整個過程如下圖示:
我們有時需要屏蔽某個信號,就需要去修改阻塞信號集。那麼,我們該如何修改阻塞信號集?系統提供的一個方法是,我們先創建一個跟阻塞信號集一樣的集合,再利用它去修改阻塞信號集。
系統提供了一系列的信號集設定函數。這些函數如下所示:
<code>sigset_t
set
; 信號集數據類型,本質是typedef
unsigned
long
sigset_t
;int
sigemptyset
(
sigset_t
*set
); 將某個信號集清0
int
sigfillset
(
sigset_t
*set
); 將某個信號集置1
int
sigaddset
(
sigset_t
*set
,int
signum); 將某個信號加入信號集int
sigdelset
(
sigset_t
*set
,int
signum); 將某個信號清出信號集 以上幾個函數返回值均是:成功:0
;失敗:-1
int
sigismember
(
const
sigset_t
*set
,int
signum); 判斷某個信號是否在信號集中 返回值:在集合:1
;不在:0
;出錯:-1
/<code>
使用以上這些函數創建完信號集後,要如何去改變阻塞信號集呢?系統又提供了一個函數:sigprocmask函數。
sigprocmask函數可以用來屏蔽信號,也可以用來解除屏蔽信號,其本質就是利用我們創建的信號集去改變阻塞信號集。
函數原型:
int sigprocmask(int how, const sigset_t set, sigset_t oldset);
返回值:
成功:0;失敗:-1,設置errno
參數解釋:
set:傳入參數,是一個位圖,set中哪位置1,就表示當前進程屏蔽哪個信號。
oldset:傳出參數,保存舊的信號屏蔽集。這個與setitimer有點相似。
how參數取值:
假設當前的信號屏蔽字為mask
1.SIG_BLOCK:當how設置為此值,set表示需要屏蔽的信號。相當於 mask = mask | set
2.SIG_UNBLOCK:當how設置為此,set表示需要解除屏蔽的信號。相當於 mask = mask & ~set
3.SIG_SETMASK:當how設置為此,set表示用於替代原始屏蔽及的新屏蔽集。相當於 mask = set若,調用sigprocmask解除了對當前若干個信號的阻塞,則在sigprocmask返回前,至少將其中一個信號遞達。
我們如何讀取未決信號集?系統提供了sigpending函數。
函數原型:
int sigpending(sigset_t *set);
參數說明:
set傳出參數。
返回值:
返回值:成功:0;失敗:-1,設置errno
例:把所有常規信號的未決狀態打印至屏幕。
<code>1
2
3
4
5void
printPending(sigset_t
*set)
6
{
7
int
i
=
0
;
8
9
for
(i
=
0
;
i
<
32
;
i++)
{
10
if
(sigismember(set,
i)
==
1
)
11
printf("1");
12
else
13
printf("0");
14
}
15
printf("\n");
16
}
17
18int
main()
19
{
20
sigset_t
set,
oldset,
pendset;
21
sigemptyset(&set);
22
sigaddset(&set,
SIGQUIT);
//
ctrl
+
\
將產生SIGQUIT信號
23
sigprocmask(SIG_BLOCK,
&set,
&oldset);
24
while
(1)
{
25
sigpending(&pendset);
26
printPending(&pendset);
//
寫一個函數打印未決信號集
27
sleep(1);
28
}
29
}
/<code>
> 2020 精選 阿里/騰訊等一線大廠 面試、簡歷、進階、電子書 「**良許Linux**」後臺回覆「**資料**」免費獲取
#### 看完的都是真愛,點個贊再走唄?您的「三連」就是良許持續創作的最大動力!
1. 關注**原創**「**良許Linux**」,第一時間獲取最新Linux乾貨!
2. 後臺回覆【資料】【面試】【簡歷】獲取精選一線大廠面試、自我提升、簡歷等資料。
3. 關注我的博客:[lxlinux.net](http://www.lxlinux.net)