一个用户进程常常需要对多个信号进行处理,为了方便对多个信号进行处理,在linux系统中引入了信号集,信号集是用来表示多个信号的数据类型
sigset_t
sigemptyset
sigfillset
sigismember
sigaddset
sigdelset
sigemptyset函数
功能: 初始化一个空的信号集
#include <signal.h> int sigemptyset(sigset_t * set)
功能
初始化由set指向的信号集,清除其中所有的信号即初始化一个空信号集
参数
set:信号集标识的地址,以后操作此信号集,对set进行操作就可以了
返回值
成功返回 0, 失败返回-1
sigfillset函数
功能:初始化一个满的信号集
#include <signal.h> int sigfillset(sigset_t * set);
功能
初始化信号集合set,将信号集合设置为所有信号的集合
参数
信号集标识的地址,以后操作此信号集,对set进行操作就可以了
返回值
成功返回0,失败返回-1
sigismember函数
判断某个集合中是否有某个信号
#include <signal.h> int sigismember(const sigset_t * set, int signum);
功能
查询signum标识的信号是否在信号集合set之中
参数
set: 信号集标识符号的地址4
signum: 信号的编号
返回值
在信号集中返回-1,不在信号集中返回0
错误,返回-1
sigaddset函数
向某个集合中添加一个信号
#include <signal.h> int sigaddset(sigset_t * set , int signum);
功能:
将信号signum加入到信号集合set之中
参数:
set:信号集标识的地址
signum;信号的编号
返回值:
成功返回0,失败返回-1
sigdelset函数
从某个信号集中删除一个信号
#include <signal.h> int sigdelset(sigset_t * set, int signum);
功能
将signum所标识的信号从信号集合set中删除
参数
set:信号集标识的地址
signum:信号的编号
返回值
成功:返回0
失败:返回-1
案例
#include <stdio.h> #include <signal.h> int main() { //创建一个信号集 sigset_t set; int ret = 0; //初始化一个空的信号集 sigemptyset(&set); //判断SIGINT信号是否在信号集中 ret = sigismember(&set,SIGINT); if(ret == 0) { printf("SIGINT is not a member of sigprocmask \n ret = %d\n", ret); } //将指定的信号添加到信号集中 sigaddset(&set, SIGINT); sigaddset(&set, SIGQUIT); //判断SIGINT是否在信号集中 ret = sigismember(&set, SIGINT); if(ret == 1) { printf("SIGINT is a member of sigprocmask \nret = %d\n", ret); } return 0; }
每个进程都有一个阻塞集,它用来描述哪些信号递送到该进程的时候被阻塞(在信号发生时记住它,直到进程准备好再将信号通知进程)
所谓阻塞并不是禁止传送信号,而是暂缓信号的传送,若将被阻塞的信号从信号阻塞集中删除,且对应的信号在被阻塞时发生了,进程将会收到相应的信号
sigprocmask函数
创建一个阻塞集合
#include <signal.h> int sigprocmask(int how, const sigset_t* set, sigset_t* oldset);
功能
检查或修改信号阻塞集,根据how指定的方法对进程的阻塞集合进行修改,新的信号阻塞集由set指定,而原先的信号阻塞集由oldset保存
参数
how:信号阻塞集合的修改方法
set: 要操作的信号集地址
保存原先信号集地址
how:
SIG_BLOCK; 向信号阻塞集合中添加set信号集
SIG_UNBLOCK; 从信号阻塞集合中删除set集合
SIG_SETMASK; 将信号阻塞集合设为set集合
注:若set为NULL,则不改变信号阻塞集合,函数只把当前信号阻塞集合保存到oldset中
返回值
成功:返回0
失败:返回-1
#include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <signal.h> int main() { int i = 0; //创建信号集,并在信号集中添加信号 sigset_t set; sigemptyset(&set); sigaddset(&set, SIGINT); while(1) { //将set信号集添加到信号阻塞集中 sigprocmask(SIG_BLOCK, &set, NULL); for(i=0; i<5; i++) { printf("SIGINT signal is blocked\n"); sleep(1); } //将set信号集从信号阻塞集中删除 sigprocmask(SIG_UNBLOCK, &set, NULL); for(i=0; i<5; i++) { printf("SIGINT signal unblocked\n"); sleep(1); } } return 0; }
我们可以看出五秒结束之后,立即运行set里面的信号SIGINT
一般情况下信号阻塞集,都会跟信号集一块来玩,因为信号阻塞集里面就是添加的是信号集,而不是某一个信号