几个基本概念
信号未决:从信号的产生到信号递达的这段时间中的状态,称为信号未决信号递达:收到信号,进行信号的处理
信号阻塞:阻塞一种信号后,这种信号将永不递达;只有解除阻塞才可以递达
易错点:信号阻塞和信号忽略的概念
信号忽略指的是收到了信号,但是不进行任何处理。而信号阻塞是根本收不到信号。信号阻塞表、信号未决表、以及自定义函数表
如何处理在解除阻塞前产生的多次信号
普通信号在递达前产生了多次,但是只计算一次;而实时信号在递达之前产生多次可以依次放在一个队列内部。
未决和阻塞标志可以用相同的数据类型sigset_t来存储,sigset_t称为信号集
这个类型可以表示每个信号的“有效”或“无效”状态
在阻塞信号集中“有效”和“无效”的含义是该信号是否被阻塞,而在未决信号集中“有效”和“无效”的含义是该信号是否处于未决状态。
阻塞信号集也叫做当前进程的信号屏蔽字(Signal Mask)
信号集和信号量集的误区
他们是没有关系的。当利用信号量机制解决了单个资源的互斥访问后,我们讨论如何控制同时需要多个资源的互斥访问,信号量集是指同时需要多个资源时的信号量操作。
信号集处理函数
初始化set所指向的信号集,使所有位 置为0
int sigemptyset(size_t * set);
初始化Set所指向的信号量集,使所有位 置为1
int sigfillset(size_t set);在信号集中添加某个有效信号
int sigaddset(size_t set, int signo);
在信号集中删除某个有效信号
int sigdelset(size_t set,int signo);
判断一个set信号集中有没有包含signo信号
int sigismember(const sigset_t * set,int signo);
若有返回1,没有返回0,出错返回-1
获取或者更改阻塞信号集
sigpromask(int how,const sigset_t *set, sigset_t *oset);
参数
oset:如果oset是非空指针,则读取进程的当前信号屏蔽字通过oset参数传出
如果set是非空指针,则更改进程的信号屏蔽字,参数how指示如何更改。
set:
如果oset和set都是非空指针,则先将原来的信号屏蔽字备份到oset里,然后根据set和how参数更改信号屏蔽字。
how:
读取当前的未决信号集,并用set返回
sigpending(sigset_t * set);成功返回0,出错返回-1
测试代码
#include<stdio.h>
#include<signal.h>
#include<unistd.h>
#include<sys/types.h>
void myhandler(int signal)
{
printf("当前进程ID为: %d ,收到 %d 信号\n",getpid(),signal);
}
void showPending(sigset_t * pending)
{
int i = 1;
while(i<=31)
{
if(sigismember(pending, i) == 1)
printf("1");
else
printf("0");
++i;
}
printf("\n");
}
int main()
{
sigset_t set,oset;//定义两个信号集
sigemptyset(&set);//初始化信号集set,使所有位为0
sigemptyset(&oset);//同上
sigaddset(&set,2);//为信号集set添加2号信号
sigprocmask(SIG_SETMASK,&set,&oset);//oset保存之前的信号屏蔽字,然后将set的值设置成当前进程的信号屏蔽字
signal(2,myhandler);//进行2号信号的捕获
int count = 0;
sigset_t pending;//定义未决信号集
while(1)
{
sigpending(&pending);//获取未决信号集,保存在pengding中
showPending(&pending);//打印当前的未决信号集
sleep(1);
if(count == 3)
{
sigprocmask(SIG_SETMASK,&oset,NULL);//将阻塞信号集设置为之前保存的
count = 0;
}
count++;
}
return 0;
}
运行结果
开始没有收到信号,penging表全为0
后来收到了 由键盘的组合键ctrl+c发出的2号信号
此时,penging表的2号位置变为1
并在3秒后解除阻塞,从而递达归0
最后进程可以用Ctrl+Z来结束