有时候不希望在接到信号时就立即停止当前执行,去处理信号,同时也不希望忽略该信号,而是延时一段时间去调用信号处理函数。这种情况可以通过阻塞信号实现。
信号的阻塞概念:信号的”阻塞“是一个开关动作,指的是阻止信号被处理,但不是阻止信号产生。
信号的状态:信号递达(Delivery):实际信号执行的处理过程(3种状态:忽略,执行默认动作,
捕获)
信号未决(Pending):从产生到递达之间的状态
信号集操作函数
sigset_t set;自定义信号集。是一个32bit 64bit 128bit的数组。
sigemptyset(sigset_t *set);清空信号集
sigfillset(sigset_t *set);全部置1
sigaddset(sigset_t *set, int signum);将一个信号添加到集合中
sigdelset(sigset_t *set, int signum);将一个信号从集合中移除
sigismember(const sigset_t *set,int signum);判断一个信号是否在集合中。
设定对信号集内的信号的处理方式(阻塞或不阻塞)
#include <signal.h>
int sigprocmask( int how, const sigset_t *restrict set, sigset_t *restrict oset );(SIG_BLOCK屏蔽信号集set里面的信号,SIG_UNBLOCK解除set里面的信号的屏蔽)
返回值:若成功则返回0,若出错则返回-1
首先,若oset是非空指针,那么进程的当前信号屏蔽字通过oset返回。
其次,若set是一个非空指针,则参数how指示如何修改当前信号屏蔽字。
how可选用的值:(注意,不能阻塞SIGKILL和SIGSTOP信号)
SIG_BLOCK:把参数set中的信号添加到信号屏蔽字中
SIG_UNBLOCK:从信号屏蔽字中删除参数set中的信号
SIG_SETMASK:把信号屏蔽字设置为参数set中的信号
int pause(void);(接收不到信号是一直处于阻塞状态,直到接收到信号之后,根据信号的行为进行下一步,比如接收到的信号是终止信号,则不会运行pause下面的程序,直接终止程序)
进程一直阻塞,直到被信号中断,返回值:-1 并设置errno为EINTR
函数行为:1 如果信号的默认处理动作是终止进程,则进程终止,pause函数没有机会返回。
2如果信号的默认处理动作是忽略,进程继续处于挂起状态,pause函数不返回
3如果信号的处理动作是捕捉,则调用完信号处理函数之后,pause返回-1。
4 pause收到的信号如果被屏蔽,那么pause就不能被唤醒
#include <stdio.h>
#include <signal.h>
#include <unistd.h>
void handle(int sig) { //结构体变量执行函数
printf("this is handle\n");
}
void task() { //信号驱动程序执行函数
printf("task start\n");
sleep(3);
printf("task end\n");
}
int main() {
struct sigaction act; //创建结构体变量
act.sa_handler = handle; //定义结构体变量执行的函数
act.sa_flags = 0; //定义结构体中sa_flags的值为0
sigemptyset(&act.sa_mask);//清空sa_mask的值
sigaction(SIGINT,&act,NULL);//传递SIGINT信号
pause(); //接收信号,并根据信号行为进行下一步程序
sigset_t set,set1; //定义信号集
sigemptyset(&set); //清空信号集
sigemptyset(&set1); //清空信号集
sigaddset(&set,SIGINT); //向信号集中添加信号
sigaddset(&set,SIGHUP); //向信号集中添加信号
while(1) {
sigprocmask(SIG_BLOCK,&set);//屏蔽信号集中的信号
task();
// sigprocmask(SIG_UNBLOCK,&set);//解除屏蔽信号集中的信号
// pause(); //接收信号,如果接收不到信号一直处于阻断状态
sigsuspend(&set1); //屏蔽信号集为空,将程序挂起执行
}
while(1) {
printf("before pause\n");
sleep(1)
}
}
int sigsuspend(const sigset_t *sigmask);
功能:将进程的屏蔽字替换为由参数sigmask给出的信号集,然后挂起进程的执行
参数:sigmask:希望屏蔽的信号