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. "); 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 "); } 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,也说明了在同一时刻
产生多次同一种信号,不会对信号排队。