1.信号集基本操作
我们需要有一个能表示多个信号--信号集(signal set)的数据类型。POSIX.1定义了数据类型sigset_t以包含一个信号
集,并且定义了一下五个处理信号处理信号集函数。
#include <signal.h>
int sigemptyset(sigset_t *set); //清除set中所有的信号
int sigfillset(sigset_t *set); //使set包含所有的信号
int sigaddset(sigset_t *set, int signo); //将signo加入到set中
int sigdelset(sigset_t *set, int signo); //将signo从set中删除
//四个函数的返回值:若成功则返回0,若出错则返回-1.
int sigismemeber(const sigset_t *set, int signo);
//返回值:若真则返回1,若假则返回0,出错则返回-1.
2.sigpromask函数
调用sigpromask函数可以检测或更改其信号屏蔽字,或在一个步骤中同时执行这两个步骤。
#include <signal.h>
int sigprocmask(int how, const sigset_t *restrict set, sigset_t *restrict oset);
//返回值:成功返回0,出错返回-1.
如果oset是非空指针,那么进程的当前信号屏蔽字通过oset返回。
如果set是一个非空指正,则参数how指示如何修改当前信号屏蔽字。
如果how为0,set为NULL,而oset不为空,则在oset中返回进程当前的信号集。
下图指示了how的可选用值。
sigprocmask是仅在单线程的进程定义的,为处理多线程的进程中的信号的屏蔽,提供了另外的函数。
3.sigpending函数
sigpending函数返回信号集,其中的各个信号对于调用进程是阻塞的而不能传递,该信号集通过set参数返回。
#include <signal.h>
int sigpending(sigset_t *set);
//若成功则返回0,出错则返回-1.
实践:
#include <stdio.h>
#include <signal.h> static void sigquit(int signo){
printf("caught sigquit.\n");
if(signal(SIGQUIT, SIG_DFL) == SIG_ERR){
perror("signal");
}
} int main(void){
sigset_t newmask,oldmask,pendmask;
if(signal(SIGQUIT,sigquit) == SIG_ERR){
perror("signal");
return -1;
} sigemptyset(&newmask);
sigaddset(&newmask, SIGQUIT);
if(sigprocmask(SIG_BLOCK,&newmask,&oldmask) < 0){
perror("sigpromask");
return -1;
} sleep(5); if(sigpending(&pendmask) < 0){
perror("sigpending");
return -1;
}
if(sigismember(&pendmask,SIGQUIT)){
printf("SIGQUIT pending\n");
} if(sigprocmask(SIG_SETMASK, &oldmask, NULL) < 0){
perror("sigpromask");
return -1;
} sleep(5);
return 0;
}
运行结果:
yan@yan-vm:~/apue$ ./a.out
^\^\^\^\^\^\^\^\^\^\SIGQUIT pending
caught sigquit.
^\Quit (core dumped)
^\^\^\^\^\^\^\^\^\^\SIGQUIT pending
caught sigquit.
^\Quit (core dumped)
在程序第一次sleep时,产生了多个SIGQUIT消息,此时被pending,解除了mask后,只产生了一次action,也说明了在同一时刻
产生多次同一种信号,不会对信号排队。