【Linux】信号的阻塞和递达

来源:互联网 发布:易语言发送qq消息源码 编辑:程序博客网 时间:2024/06/06 01:13

几个基本概念

信号未决:从信号的产生到信号递达的这段时间中的状态,称为信号未决
信号递达:收到信号,进行信号的处理

信号阻塞:阻塞一种信号后,这种信号将永不递达;只有解除阻塞才可以递达

易错点:信号阻塞和信号忽略的概念

信号忽略指的是收到了信号,但是不进行任何处理。而信号阻塞是根本收不到信号。

信号阻塞表、信号未决表、以及自定义函数表


如何处理在解除阻塞前产生的多次信号

普通信号在递达前产生了多次,但是只计算一次;而实时信号在递达之前产生多次可以依次放在一个队列内部。

未决和阻塞标志可以用相同的数据类型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");elseprintf("0");++i;}printf("\n");}int main(){sigset_t set,oset;//定义两个信号集sigemptyset(&set);//初始化信号集set,使所有位为0sigemptyset(&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来结束

原创粉丝点击