【Linux系统编程二十三】:(信号2)--信号的保存

【Linux系统编程二十三】:信号的保存

  • 一.信号的保存
    • 1.阻塞信号
    • 2.sigset_t类型(位图)
    • 3.block表
    • 4.handler表
    • 5.pending表
  • 二.实验验证
  • 三.信号的其他概念

一.信号的保存

信号发送本质上是操作系统发送信号,而进程PCB内部有一个位图用来表示是否接收到信号。如果接收到某个信号,该位置上的比特位就会由0置1。该位图我们称为penging表。

信号本质上其实就是一个一到三十一的数字,并且每一种信号都要有它对应的处理方法,即每一个信号都有它自己默认处理方法。所以进程的内核当中呢,还要为每一个信号维护一个叫做handler表。这个handle表它的类型是一个函数指针数组。里面都是信号对应默认处理方法。这里面的默认处理方法可以被捕捉变成自定义方法。
在这里插入图片描述

1.这个信号将来可以充当我们数组的下标,找到某一个位置。所以每个信号都对应着函数指针数组里的方法。
2.所以一旦信号产生了,那么我们直接拿着信号编号缩影一张表,就可以找到处理该信号的方式。

我们为什么要保存信号呢?

因为进程接收到信号之后,可能不会立即处理这个信号,信号不会被立即处理,那么就要有一个时间窗口,信号需要保存起来。

所以我们进程在运行时,一旦接收到信号了,那么对应的信号位图上的比特位就会由0置1,进程会处理这个信号,但是在合适的时候去处理。
所以信号暂时就在位图中保存起来。等进程忙完它的事情后,回来根据位图上的比特位来是否处理。这样信号就保存起来了。

不过信号的保存,不单单是信号位图的保存记录,还有信号阻塞,信号方法的保存。

1.阻塞信号

其实进程可以选择阻塞某个信号。什么意思呢?
进程接收到诸多信号后,可能不会立即处理,而是在适当的时候处理,但最后总归是会处理的。而如果一旦你把某些信号屏蔽了,在该信号没有被解除屏蔽之前,那么即便你收到了该信号。那么对应的信号也不会被操作系统系统进行处理。
还有如果没有产生一个信号,可以被屏蔽吗?可以被阻塞吗?
当然可以。屏蔽是一种状态。和你当前是否产生没关系。
所以在内核里进程还需要维护一张表叫做block表。它也是一张位图,比特位的位置决定的是信号的编号,比特位的内容是为0表示不屏蔽,为1表示屏蔽。那么阻塞也就意味着信号不会被处理。

在这里插入图片描述
理解普通信号时,我们只需要了解清楚三张表就可以,分别叫做pending表、block表和handler表。
pending表用来记录当前进程是否收到了信号,以及收到了哪些信号。
block表用来记录特定信号,那么是否被屏蔽。
handler表描述的是每一种信号所对的处理关系。

所以呢我们有了三张表,即两张位图和一个函数指针数组,就可以用来实现对普通信号的记录,保存相关的管理工作。

2.sigset_t类型(位图)

刚刚说的那pending表、handler表、block它都属于操作系统内部的内核数据结构。操作系统不相信任何用户,他不允许用户直接去修改。
如果用户你想改甚至想读获取这些表,对不起,不能直接读。
所以我们一定未来的访问这三张表是必须得有系统调用接口。

我们要获取这几个位图的时候,那么就注定了要在用户空间和内核空间内核空间和用户空间进行来回的数据拷贝。
所以数据拷贝时,我们就要在接口的参数设计上,要设置我们对应的输出出行参数。理论上获取内核里位图的信息,我们应该也用一个位图变量,将内核里的数据都拷贝到这个位图变量里来,然后由这个位图变量带出来。

如果用户想拿到你这个进程的pending表或block表,那么他怎么获取呢?
所以就必须要求操作系统给我们在应用上设计出一种数据类型。那么这个sigset_t类型呢,它就是我们传说中的位图结构。这是操作系统呢给我们提供了一种类型,叫做信号集类型。它是由操作系统给我们提供,在我们用户层可以直接使用的一种数据类型。

它属于系统级的数据类型,那么我们内部如何去呃存储对应的比特位呢?我们是不能直接用位操作获取位图里的比特位的!
在这里插入图片描述

如果我们将来拿到了进程的penging表、block表,以及我们拿到了我们所谓的sigset_t这样的位图类型,你也绝对不能自己再来进行位操作了啊,不允许你这么做,因为这是操作系统级别的类型。用户不能直接操作。

我们如果要修改对应的这这个sigset_t 位图,我们需要使用下面的这批系统调用。
在这里插入图片描述

3.block表

在这里插入图片描述
在这里插入图片描述
sigprocmask是用来读取或者更改进程的屏蔽信号的
它等价于其实就是阻塞信号器。它可以通过我们传递三个参数来达到对信号屏蔽字。

1.int how 第一个参数:是对block表的操作,你要干啥,是要新增一个屏蔽字还是删除一个屏蔽字呢?总共有三种选择:
①SIG_BLOCK:新增一个屏蔽字
②SIG_UNBLOCK:解除一个屏蔽字
③SIG_SETMAS:覆盖式的设置屏蔽字
2.sigset_t *set 第二个参数:是用来设置屏蔽字的,你要新增或者屏蔽哪个字段都在这个位图里设置,然后它会作为输入型参数配合how,设置到内核里。
3.sigset_t *oset 第三个参数:是将内核里的block位图先保存起来,未来可以还需要使用,保存就是为了恢复。所以作为输出型参数。

4.handler表

handler表本质上就是一个函数指针数组,它里面是信号对应的默认处理方法或者自定义方法。我们利用捕捉就可以获取到handler表里的方法。在这里插入图片描述
捕捉的方法有两种,一个是signal,一个是sigaction。
signal有两个参数,一个参数signum是信号的编号,另一个sighandler_t handler参数就是信号对应的处理方法。可以是操作系统提供的默认方法,也可以是自定义方法。我们应该在一开始就设置捕捉,而不是在最后,因为,捕捉信号只有接收到信号才会捕捉,没有接收到信号是不会捕捉的。如果放在最后,万一接收信号的时候,还没执行到捕捉代码那就不行了。所以捕捉代码最好写在一开头。

捕捉的本质就是去执行信号对应的方法。

在这里插入图片描述

5.pending表

在这里插入图片描述
这个sigpending函数就是用来获取pending位图里的比特位信息的。
它的参数是一个输出型参数,它的作用非常简单,它的作用就是要帮助我们把。调用进程它所对应的pending表给我们以位图的形式带出来。

二.实验验证

我们可以做一个验证:将二号信号屏蔽,然后发送二号信号,捕捉二号信号,再获取进程的pending表,最后再解除二号信号的屏蔽。会发生什么样的过程呢?

在这里插入图片描述

#include <iostream>
#include <signal.h>
#include <unistd.h>
using namespace std;void PrintPending()
{sigset_t set;//位图变量,输出型参数sigpending(&set);//将penging表带出来for(int signo=1;signo<=31;signo++){if(sigismember(&set,signo))//如果signo信号在set表里就为真cout<<"1";elsecout<<"0";}cout<<endl;}void handler(int signo)
{cout<<"catch a signal "<<signo<<endl;
}
//保存信号主要靠三张表---block表,pending表和handler表//现在我们想主动的屏蔽一个信号该如何实现呢?利用sigprocmask系统调用int main()
{//0.首先对2号信号进行捕捉signal(2,handler);//1.对二号信号进行屏蔽//首先要信号是在位图里,所以我们需要利用位图来修改信号,需要位图变量。sigset_t bset,oset;//输入型参数,输出型参数,保存就是为了恢复sigemptyset(&bset);//对这两个位图变量初始化sigemptyset(&oset);sigaddset(&bset,2);//设置屏蔽字,将二号信号添加到bset位图里sigprocmask(SIG_SETMASK,&bset,&oset);//将best覆盖式屏蔽位图里的信号int cnt=0;//2.重复打印当前进程的pending表while(true){PrintPending();sleep(1);cnt++;if(cnt==10)//解除2好信号屏蔽{cout<<"unblock signal"<<endl;sigprocmask(SIG_SETMASK,&oset,nullptr);//将原来的旧表覆盖}}
}

它的结果就是当发送2号信号时,进程并不会捕捉2号信号处理,打印pending表示,显示2号信号上的位图由0置1,说明接收到信号。但不处理。说明被屏蔽。
当解除屏蔽时,进程就会立即处理2号信号。

还有一件事,那就是并不是所有信号都可以被屏蔽,19号信号就不能捕捉也不能屏蔽。
在这里插入图片描述

所以信号最重要的就是这三个表了。而对这三个表的操作也很重要。
在这里插入图片描述

三.信号的其他概念

在这里插入图片描述
1.信号递达对应处理的是handler表。信号未决由pending表显示。信号阻塞由block表显示。
2.普通信号只会保存一次,即如果一次性发送许多同一的信号,该信号只会被保存一次。而如果该信号被屏蔽了,当发送许多相同信号,pending位图里只会记录一次。当该信号解除屏蔽,也只会执行一次方法。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/298454.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

WGCLOUD监控系统的server和agent作用详解

WGCLOUD包括&#xff1a;server为服务端&#xff08;或主控端&#xff09;&#xff0c;agent为客户端&#xff08;探针端、被控端&#xff09; WGCLOUD是绿色版本&#xff0c;非侵入式&#xff0c;解压即可运行&#xff0c;是完全自主私有化部署的监控平台&#xff0c;不依赖外…

HTTP分数排行榜

HTTP分数排行榜 介绍一、创建数据库二、创建PHP脚本三、上传下载分数四、测试 介绍 Unity中向服务器发送用户名和得分&#xff0c;并存入数据库&#xff0c;再讲数据库中的得分按照降序的方式下载到Unity中。 一、创建数据库 首先&#xff0c;我们要在MySQL数据库中建立一个…

《Spring Cloud学习笔记:Nacos配置管理 OpenFeign Getway》

基于Feign的声明式远程调用&#xff08;代码更优雅&#xff09;&#xff0c;用它来去代替我们之前的RestTemplate方式的远程调用 1. Nacos配置管理 Nacos除了可以做注册中心&#xff0c;同样也可以做配置管理来使用。 利用Nacos实现统一配置管理以及配置的热更新&#xff1a;…

微服务-springcloud(eureka实践, nacos实践)

Spring 体系图 版本关系 eureka 实践 1 父工程依赖 <parent><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-parent</artifactId><version>2.6.14</version> </parent> <dependencyManage…

matplotlib 自定义颜色

使用在线取色器获取RGB信息&#xff0c;比如(68, 114, 197)&#xff0c;传入归一化后的RGB颜色color(68/255, 114/255, 197/255)即可。 import matplotlib.pyplot as plt import matplotlib as mpl # 中文和负号的正常显示 mpl.rcParams[font.sans-serif] [Times New Roman] …

Python 数据分析 Matplotlib篇 plot设置线条样式(第2讲)

Python 数据分析 Matplotlib篇 plot设置线条样式(第2讲)         🍹博主 侯小啾 感谢您的支持与信赖。☀️ 🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ�…

算法学习——二叉树

二叉树 二叉树理论基础满二叉树完全二叉树二叉搜索树平衡二叉搜索树二叉树的存储方式二叉树的遍历方式二叉树的定义 二叉树的递归遍历思路 二叉树的迭代遍历思路前序遍历后序遍历中序遍历 二叉树的层序遍历二叉树的层序遍历思路代码 二叉树的层序遍历II思路代码 二叉树的右视图…

RT-Smart 官方 ARM 32 平台 musl gcc 工具链下载

前言 RT-Smart 的开发离不开 musl gcc 工具链&#xff0c;用于编译 RT-Smart 内核与用户态应用程序 RT-Smart musl gcc 工具链代码当前未开源&#xff0c;但可以下载到 RT-Thread 官方编译好的最新的 musl gcc 工具链 ARM 32位 平台 比如 RT-Smart 最好用的 ARM32 位 qemu 平…

18.仿简道云公式函数实战-数学函数-AVERAGE

1. AVERAGE函数 AVERAGE 函数可用于计算一组数值的算术平均值。 2. 函数用法 AVERAGE(数字1,数字2,...) 3. 函数示例 AVERAGE(1,3,5)&#xff0c;返回结果为 3 4. 代码实战 首先我们在function包下创建math包&#xff0c;在math包下创建AvgFunction类&#xff0c;代码如…

通过字符设备驱动点亮板子上的led灯

通过字符设备驱动点亮板子上的led灯 app: test.c char buf[3] 1 0 0 0 1 0 0 0 1 ------------------|------------------------ kernel: led_driver.c -------------------|------------------------ hardware: RGB_led 应用程序如何将数据传递给驱动&#xff08;读写…

工程数学软件:PTC Mathcad Prime 9.0 Crack

PTC Mathcad Prime 是工程数学软件的行业标准&#xff0c;使您能够解决最复杂的问题并共享您的工程计算。借助 PTC Mathcad Prime 9&#xff0c;工程计算变得更加出色。此版本引入了关键应用程序、符号引擎、数字引擎和可用性增强功能。 可用性和生产力更新        Mathc…

制作一个可以离线安装的Visual Studio安装包

须知 前提条件&#xff0c;需要电脑可以正常上网且网速还行&#xff0c;硬盘可以空间容量足够大&#xff0c;怎么判断容量够用&#xff1f;由组件数量的多少来决定。Visual Studio 频道和发布节奏 https://learn.microsoft.com/zh-cn/visualstudio/productinfo/release-rhythm…