讲到pending,就不得不提一下阻塞信号方面的知识:
实际执行信号的处理动作称为信号递达(Delivery),信号从产生到递达之间的状态,称为信号未(Pending)。进程可以选择阻塞(Block )某个信号。被阻塞的信号产生时将保持在未决状态,直到进程解除对此信号的阻塞,才 执行递达的动作。
注意:
阻塞和忽略是不同的,只要信号被阻塞就不会递达,而忽略是在递达之后可选的一种处理动作
信号间三张表的关系(block->pending->handler)
那么信号是怎么产生的呢?
1、键盘组合键(Ctrl+C、Ctrl+\、Ctrl+Z……)
2、软硬件产生(指针异常,内存溢出)
3、kill 信号(Linux下kill -l可对信号进行查看)
那收到信号了,又该怎么处理应对呢?
处理方式,一般有三种:
1.忽略
2.默认动作
3.用户自定义
信号阻塞:
信号阻塞,这PCB中,block所对应的比特位是“1”,表示信号阻塞,pending未决状态位可为“1”或者“0”,如果信号一旦递达,则进行信号处理(handler)
信号集操作函数
sigset_t类型对于每种信号用⼀一个bit表示“有效”或“无效”状态
int sigemptyset(sigset_t *set);//初始化set所指向的信号集,使其中所有信号的对应bit清零,表示该信号集不包含任何有效信号
int sigfillset(sigset_t *set);//t初始化set所指向的信号集,使其中所有信号的对应bit置位有效,表示该信号集的有效信号包括系统支持的所有信号
int sigaddset(sigset_t *set, int signo);//信号集中添加某种有效信号
int sigdelset(sigset_t *set, int signo);//信号集中删除某种有效信号
int sigismember(const sigset_t *set, int signo); //用于判断一个信号集的有效信号中是否包含某种信号
调用函数sigprocmask可以读取或更改进程的信号屏蔽字(阻塞信号集)。
int sigprocmask(int how, const sigset_t *set, sigset_t *oset);
返回值:若成功则为0,若出错则为-1
how参数:
1)SIG_BLOCK添加到当前信号屏蔽字的信号
2)SIG_UNBLOCK从当前信号屏蔽字中解除的信号
3)SIG_SETMASK设置当前信号屏蔽字为set所指向的值
#include<stdio.h>
#include<signal.h>
#include<unistd.h>
void printsigset(sigset_t *set)
{
int i = 0;
for(;i<32;i++)
{
if(sigismember(set,i))
{
putchar('1');
}
else
{
putchar('0');
}
}
puts("");
}
int main()
{
sigset_t s,p;
sigemptyset(&s);
sigaddset(&s,SIGINT);
sigprocmask(SIG_BLOCK,&s,NULL);
while(1)
{
sigpending(&p);
printsigset(&p);
sleep(1);
}
return 0;
}
Ctrl+C前后
前:
后: