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的可选用值。
《UNIX环境高级编程》笔记--信号集_单线程
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)
在程序第一次sleep时,产生了多个SIGQUIT消息,此时被pending,解除了mask后,只产生了一次action,也说明了在同一时刻
产生多次同一种信号,不会对信号排队。