linux环境编程-IPC【signal-信号集操作函数】

前面我们介绍了能产生信号的函数,以及他们的传递信号给那些谁【非常的详细了】

今天来介绍一下 信号集的操作函数。仔细看下面的视频,别眨眼,你会喜欢的。

一.信号集操作函数
内核通过读取未决信号集来判断信号是否应被处理。信号屏蔽字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是不能阻塞的】

 

 

 

 

 

 

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值