信号集给我们提供了一个能表示多个信号的是数据类型(sigset_t), 它将在sigprocmask, sigpending, sigsuspend之类的函数中用到, 这些函数我会在以后的文章中介绍.
1. 信号集相关函数:
#include
int sigemptyset(sigset_t *set);
成功则返回0, 出错则返回-1.
这个函数用作初始化set指向的信号集, 清空其中的所有信号.
#include
int sigfillset(sigset_t *set);
成功则返回0, 出错则返回-1.
这个函数用作初始化set指向的信号集, 填充其中的所有信号.
#include
int sigaddset(sigset_t *set, int signo);
int sigdelset(sigset_t *set, int signo);
成功则返回0, 出错则返回-1.
这两个函数用作向set指向的信号集中, 增加/删除一个signo代表的信号.
#include
int sigismember(const sigset_t *set, int signo);
真则返回1, 假则返回0, 出错则返回-1.
这个函数用作判断signo信号是否在set指向的信号集中.
2. 宏:
在signal.h中有两个宏:
#define sigemptyset(ptr) (*(ptr) = 0)
#define sigfillset(ptr) (*ptr = ~(sigset_t)0, 0)
这两个宏分别定义了sigemptyset和sigfillset两个函数的行为.
sigemptyset: 把ptr指向的地址的内容设为0.
sigfillset: 这是一个逗号表达式, 把0转换为sigset_t类型, 然后按位取反, 并返回0.
通过以上两个宏, 我们可以确切地说, sigset_t是用多位(比信号总数更多的位数)来表示信号集概念的.
因此,
sigaddset: 将某一位设置为1.
sigdelset: 将某一位设置为0.
sigismember: 测试某一个指定位.
下面我们来实现这些函数.
3. 实例:
#include
#include
/* NSIG defined in */
#define SIGBAD(signo) ((signo) <= 0 || (signo) >= NSIG)
int sigaddset(sigset_t *set, int signo)
{
if (SIGBAD(signo))
{
errno = EINVAL;
return (-1);
}
*set |= 1 << (signo -1);
return 0;
}
int sigdelset(sigset_t *set, int signo)
{
if (SIGBAD(signo))
{
errno = EINVAL;
return (-1);
}
*set &= ~(1 << (signo -1));
return 0;
}
int sigismember(const sigset_t *set, int signo)
{
if (SIGBAD(signo))
{
errno = EINVAL;
return (-1);
}
return ((*set & (1 << (signo -1))) != 0);
}
说明一下里面的几个细节:
signo - 1: 因为不存在编号为0的信号, 也就是第0位与编号为1的信号是对应的, 所以减1.
SIGBAD: 小于等于0, 或者大于最大信号编号NSIG.
1 << : 1的左移位操作, 右边补0, 所以执行或操作时直接实现添加; 执行与操作时需要先取反, 以保证本位为0, 其他位不变, 这样来实现删除.
sigismember: 最后一个return语句看上去有些复杂. 想起来很难想到, 但看起来应该不难. 执行移位后的与操作, 当存在时使本位保持不变, 其他位清零; 当不存在时, 全部清零. 然后判断是否为0, 这样来实现判断存在性.
阅读(1100) | 评论(0) | 转发(0) |