前面我们介绍了能产生信号的函数,以及他们的传递信号给那些谁【非常的详细了】
今天来介绍一下 信号集的操作函数。仔细看下面的视频,别眨眼,你会喜欢的。
一.信号集操作函数
内核通过读取未决信号集来判断信号是否应被处理。信号屏蔽字mask可以影响未决信号集。而我们可以在应用程序中自定义set来改变mask.已达到屏蔽指定信号的目的。【在前面我已经给大家介绍过啦,怎么影响的原理】
信号集设定sigset _t set;// typedef unsigned long sigset .t; .
- int sigemptyset(sigset t *set);将某个信号集清0成功: 0;失败: -1
- int sgillset(sigset t *set);将某个信号集置1成功: 0;失败: -1
- int sigaddset(sigset t *set, int signum);将某个信号加入信号集成功: 0;失败: -1
- int sigdelset(sigset ,t *set, int signum);将某个信号清出信号集成功: 0;失败: -1
- int sigismember(const sigset _t *set, int signum);判断某个信号是否在信号集中返回值: 在集合: 1; 不在: 0;出错: -1
sigset_t 类型的本质是位图。但不应该直接使用位操作,而应该使用上述函数,保证跨系统操作有效。对比认知select 函数。
二.屏蔽某个信号
1.自定义的信号集可以用来影响PCB中的阻塞信号集(目的:把某个信号设置为阻塞(屏蔽))
2.sigprocmask函数
用来屏蔽信号、解除屏蔽也使用该函数。其本质,读取或修改进程的信号屏蔽字(PCB中)严格注意,屏蔽信号:只是将信号处理延后执行(延至解除屏蔽);而忽略表示将信号丢处理。
int sigprocmask(int how, const sigset t *set, sigset t *oldset);成功: 0; 失败: -1, 设置errno
参数:
- set:传入参数,是一个位图,set 中哪位置1,就表示当前进程屏蔽哪个信号。
- oldset:传出参数,保存旧的信号屏蔽集。
- how参数取值:假设当 前的信号屏蔽字为mask
- 1. SIG_ BLOCK: 当how设置为此值,set表示需要屏蔽的信号。相当于mask = mask|set
- 2. SIG_ UNBLOCK: 当how设置为此,set表示需要解除屏蔽的信号。相当于mask = mask & ~set
- 3. SIG_ SETMASK: 当how设置为此,set表示用于替代原始屏蔽及的新屏蔽集。相当于mask= set
若,调用sigprocmask解除了对当前若干个信号的阻塞,则在sigprocmask返回前,至少将其中一个信号递达。
2.sigpending函数:
读取当前进程的未决信号集int sigpending(sigset t *set); set 传出参数。返回值:成功: 0;失败: -1,设置errmo
三.代码Demo【正好我们验证一下(前面提及的)SIGKILL 能不能被阻塞】
#include <iostream>
#include <signal.h>
#include <unistd.h>
using namespace std;
static int Debug = 1;
void
printped(sigset_t* ped)
{
int i;
int ret;
if(Debug)printf("come here\n");
for(i=1; i<32; i++){
ret = sigismember(ped, i);
if(ret == 1){
cout<<1;
}else if(ret == 0){
cout<<0;
}else{
perror("sigismember error");
return;
}
}
return ;
}
int
main(int argc, char*argv[])
{
sigset_t old_set, new_set, ped;
sigemptyset(&new_set);
sigaddset(&new_set, SIGINT);
sigaddset(&new_set, SIGKILL);
sigaddset(&new_set, SIGQUIT);
// int sigprocmask(int how, const sigset_t *set, sigset_t *oldset);
sigprocmask(SIG_BLOCK, &new_set, &old_set);
while(1){
sigpending(&ped);
// if(Debug) cout << "ped: " << static_cast<unsigned long>(ped) << endl;
printped(&ped);
sleep(2);
}
return 0;
}
效果:但我们按下Ctrl + C【SIGINT在信号屏蔽字中对应的位被置1,然后在由信号屏蔽字影响未决信号集合,把对应的位置一,延迟了信号的事件处理】【由此可见SIGKILL是不能阻塞的】