Linux教程

linux进程间通信(IPC) ---- 信号集 信号阻塞集

本文主要是介绍linux进程间通信(IPC) ---- 信号集 信号阻塞集,对大家解决编程问题具有一定的参考价值,需要的程序猿们随着小编来一起学习吧!

信号集概述

一个用户进程常常需要对多个信号进行处理,为了方便对多个信号进行处理,在linux系统中引入了信号集,信号集是用来表示多个信号的数据类型

信号集数据类型

sigset_t

定义路径

/usr/include/x86_64-linux-gnu/bits/sigset.h(Ubuntu 16.04)

在这里插入图片描述

信号集相关的操作主要有如下几个函数

sigemptyset
sigfillset
sigismember
sigaddset
sigdelset

sigemptyset函数

功能: 初始化一个空的信号集

#include <signal.h>
int sigemptyset(sigset_t * set)

功能
初始化由set指向的信号集,清除其中所有的信号即初始化一个空信号集
参数
set:信号集标识的地址,以后操作此信号集,对set进行操作就可以了
返回值
成功返回 0, 失败返回-1

sigfillset函数

功能:初始化一个满的信号集

#include <signal.h>
int sigfillset(sigset_t * set);

功能
初始化信号集合set,将信号集合设置为所有信号的集合
参数
信号集标识的地址,以后操作此信号集,对set进行操作就可以了
返回值
成功返回0,失败返回-1

sigismember函数

判断某个集合中是否有某个信号

#include <signal.h>
int sigismember(const sigset_t * set, int signum);

功能
查询signum标识的信号是否在信号集合set之中
参数
set: 信号集标识符号的地址4
signum: 信号的编号
返回值
在信号集中返回-1,不在信号集中返回0
错误,返回-1

sigaddset函数

向某个集合中添加一个信号

#include <signal.h>
int sigaddset(sigset_t * set , int signum);

功能:
将信号signum加入到信号集合set之中
参数:
set:信号集标识的地址
signum;信号的编号
返回值:
成功返回0,失败返回-1

sigdelset函数

从某个信号集中删除一个信号

#include <signal.h>
int sigdelset(sigset_t * set, int signum);

功能
将signum所标识的信号从信号集合set中删除
参数
set:信号集标识的地址
signum:信号的编号
返回值
成功:返回0
失败:返回-1

案例

#include <stdio.h>
#include <signal.h>
int main()
{
		//创建一个信号集
        sigset_t set;
        int ret = 0;
		//初始化一个空的信号集
        sigemptyset(&set);
		//判断SIGINT信号是否在信号集中
        ret = sigismember(&set,SIGINT);
        if(ret == 0)
        {
                printf("SIGINT is not a member of sigprocmask \n ret = %d\n", ret);

        }
		//将指定的信号添加到信号集中
        sigaddset(&set, SIGINT);
        sigaddset(&set, SIGQUIT);
		//判断SIGINT是否在信号集中
        ret = sigismember(&set, SIGINT);
        if(ret == 1)
        {
                printf("SIGINT is a member of sigprocmask \nret = %d\n", ret);
        }
        return 0;
}       

在这里插入图片描述

信号阻塞集(屏蔽集,掩码)

每个进程都有一个阻塞集,它用来描述哪些信号递送到该进程的时候被阻塞(在信号发生时记住它,直到进程准备好再将信号通知进程)
所谓阻塞并不是禁止传送信号,而是暂缓信号的传送,若将被阻塞的信号从信号阻塞集中删除,且对应的信号在被阻塞时发生了,进程将会收到相应的信号

sigprocmask函数

创建一个阻塞集合

#include <signal.h>
int sigprocmask(int how, const sigset_t* set, sigset_t* oldset);

功能
检查或修改信号阻塞集,根据how指定的方法对进程的阻塞集合进行修改,新的信号阻塞集由set指定,而原先的信号阻塞集由oldset保存
参数
how:信号阻塞集合的修改方法
set: 要操作的信号集地址
保存原先信号集地址
how:
SIG_BLOCK; 向信号阻塞集合中添加set信号集
SIG_UNBLOCK; 从信号阻塞集合中删除set集合
SIG_SETMASK; 将信号阻塞集合设为set集合
注:若set为NULL,则不改变信号阻塞集合,函数只把当前信号阻塞集合保存到oldset中
返回值
成功:返回0
失败:返回-1

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <signal.h>
int main()
{
   		  int i = 0;
		//创建信号集,并在信号集中添加信号
        sigset_t set;
        sigemptyset(&set);
        sigaddset(&set, SIGINT);
        
        while(1)
        {
        		//将set信号集添加到信号阻塞集中
                sigprocmask(SIG_BLOCK, &set, NULL);
                for(i=0; i<5; i++)
                {
                        printf("SIGINT signal is blocked\n");
                        sleep(1);
                }
                //将set信号集从信号阻塞集中删除
				sigprocmask(SIG_UNBLOCK, &set, NULL);
				for(i=0; i<5; i++)
				{
					printf("SIGINT signal unblocked\n");
					sleep(1);
				}
        }
        return 0;
}

我们可以看出五秒结束之后,立即运行set里面的信号SIGINT
一般情况下信号阻塞集,都会跟信号集一块来玩,因为信号阻塞集里面就是添加的是信号集,而不是某一个信号
在这里插入图片描述

这篇关于linux进程间通信(IPC) ---- 信号集 信号阻塞集的文章就介绍到这儿,希望我们推荐的文章对大家有所帮助,也希望大家多多支持为之网!