进程间通信5:信号

news2025/1/13 11:48:41

引入

我们之前学习了信号量,信号量和信号可不是一个东西,不能混淆。

信号是什么以及一些基础概念

信号是一种让进程给其他进程发送异步消息的方式

  1. 信号是随时产生的,无法预测
  2. 信号可以临时保存下来,之后再处理
  3. 信号是异步发送的。因为这两个进程(发送信号的进程和接收信号的进程)互不相干

kill -l :查看信号

我们可以使用kill -l查看所有信号

他的输出大概是这样子

1) SIGHUP      2) SIGINT      3) SIGQUIT     4) SIGILL
5) SIGTRAP     6) SIGABRT     7) SIGBUS      8) SIGFPE
...
  1. 可以发现没有0、32、33号信号。1-31分别对应一个bit位
  2. 34到64号信号是实时信号:当开始执行实时信号,必须执行完才能执行其他的信号

信号处理

面对信号,我们有多种处理方式:

  1. 默认动作

  2. 自定义处理–>捕捉

  3. 忽略信号

我们就是通过signal系统调用来更改处理信号的方式

信号的产生

有三种方式:kill命令、键盘输入、系统调用

kill命令

使用kill命令

kill -num pid

常见的就是

kill -9 pid
#终止进程

键盘输入

像是输入ctrl c也可以停止当前进程

使用系统调用函数

  1. 使用kill函数:给指定的进程发送指定的信号
#include <sys/types.h>
#include <signal.h>

int kill(pid_t pid, int sig);

成功返回0,失败返回-1,并设置errno

  1. raise函数:对调用raise的进程发送信号
#include <signal.h>

int raise(int sig);

这个函数功能相当于调用

kill(getpid(), sig);
  1. abort函数
    调用该函数的进程直接退出

  2. 使用signal:修改信号

使用typedef简化写法:

typedef void (*signal_handler_t)(int);
signal_handler_t signal(int sig, signal_handler_t func);
//func是回调函数
//底层调用func的时候,func的参数就是sig

我们先传入要对哪个信号进行修改,再传入我们自定义的修改方法(func)
并且修改一次后一直生效

SIG_IGN
是ignore,忽略信号的意思,

signal(num, SIG_IGN)
//接收num信号后,不执行任何操作
//9号19号信号无法被忽略(可以区了解一下这两个信号的作用就能理解为什么了

异常

最常见的就是代码出现问题,爆出了异常

比如出现num/0的情况,产生SIGFPE信号,
访问野指针,产生11号信号SIGSEGV

Core Dump

  1. 是什么
    当一个进程要异常终止时,可以选择**把进程的用户空间内存数据全部保存到磁盘上,文件名通常是core,**这叫做Core Dump。

  2. 为什么
    进程异常终止通常是因为有Bug,比如非法内存访问导致段错误,事后可以用调试器检查core文件以查清错误原因,这叫做Post-mortem Debug(事后调试)。一个进程允许产生多大的core文件取决于进程的Resource Limit(这个信息保存 在PCB中)。默认是不允许产生core文件的, 因为core文件中可能包含用户密码等敏感信息,不安全,以及防止频繁崩溃导致生成大量core dump文件
    在开发调试阶段可以用ulimit命令改变这个限制,允许产生core文件。 首先用ulimit命令改变Shell进程的Resource Limit,允许core文件最大为1024K:

ulimit -c 1024

信号保存

信号其他相关常见概念

  1. 实际执行信号的处理动作称为信号递达(Delivery) (默认、忽略、自定义)
  2. 信号从产生到递达之间的状态,称为信号未决(Pending)(信号被临时保存)
  3. 进程可以选择阻塞 (Block )某个信号。(即无法执行该信号,具体原理是位图)
  4. 被阻塞的信号产生时将保持在未决状态,直到进程解除对此信号的阻塞,才执行递达的动作.
  5. 注意,阻塞和忽略是不同的,只要信号被阻塞就不会递达,而忽略是在递达之后可选的一种处理动作
  6. 9和19号进程无法屏蔽

内核中的结构:三张表(重要)

在这里插入图片描述这张图显示了block位图记录信号是否阻塞、pending位图表示未决、handler表示函数指针数组,记录信号执行方法

对三张表的操作

从上图来看,每个信号只有一个bit的未决标志,非0即1,不记录该信号产生了多少次,阻塞标志也是这样表示的。
因此,未决和阻塞标志可以用相同的数据类型sigset_t来存储,sigset_t称为信号集,这个类型可以表示每个信号的“有效”或“无效”状态,在阻塞信号集中“有效”和“无效”的含义是该信号是否被阻塞,而在未决信号集中“有效”和“无效”的含义是该信号是否处于未决状态。
阻塞信号集也叫做当前进程的信号屏蔽字(Signal Mask),这里的“屏蔽”应该理解为阻塞而不是忽略。

基本接口(了解即可

#include <signal.h>

// 初始化信号集为空(不包含任何信号)。
// 参数:
//   set - 指向要初始化的信号集。
// 返回值:
//   成功返回 0,失败返回 -1。
int sigemptyset(sigset_t *set);

// 将信号集中的所有信号置为有效(包含所有信号)。
// 参数:
//   set - 指向要填充的信号集。
// 返回值:
//   成功返回 0,失败返回 -1。
int sigfillset(sigset_t *set);

// 向信号集中添加一个指定的信号。
// 参数:
//   set   - 指向信号集。
//   signo - 要添加的信号编号。
// 返回值:
//   成功返回 0,失败返回 -1。
int sigaddset(sigset_t *set, int signo);

// 从信号集中删除一个指定的信号。
// 参数:
//   set   - 指向信号集。
//   signo - 要删除的信号编号。
// 返回值:
//   成功返回 0,失败返回 -1。
int sigdelset(sigset_t *set, int signo);

// 检查某个信号是否在信号集中。
// 参数:
//   set   - 指向信号集。
//   signo - 要检查的信号编号。
// 返回值:
//   如果信号在信号集中,返回 1;否则返回 0;失败返回 -1。
int sigismember(const sigset_t *set, int signo);

在使用sigset_ t类型的变量之前,一定要调 用sigemptyset或sigfillset做初始化,使信号集处于确定的状态。初始化sigset_t变量之后就可以在调用sigaddset和sigdelset在该信号集中添加或删除某种有效信号。

注意: 上面的接口都没有写入内核中,需要

sigprocmask:修改block位图

#include <signal.h>
int sigprocmask(int how, const sigset_t *set, sigset_t *oset);
返回值:若成功则为0,若出错则为-1

在这里插入图片描述

sigpending:查看未决信号集

#include <signal.h>

int sigpending(sigset_t *set);

可以用于 查看哪些信号被屏蔽且处于等待状态。

补充
pending位图先清零,对应信号再递达

信号的处理

什么时候处理

进程从内核态切换到用户态时,OS检测并处理信号

如何处理

可以先看一下下面这个图
在这里插入图片描述

能看到有4次状态切换,这里的原理是:
如果信号的处理动作是用户自定义函数,在信号递达时就调用这个函数,这称为捕捉信号。
由于信号处理函数的代码是在用户空间的,处理过程比较复杂,举例如下: 用户程序注册了SIGQUIT信号的处理函数sighandler。 当前正在执行main函数,这时发生中断或异常切换到内核态。 在中断处理完毕后要返回用户态的main函数之前检查到有信号SIGQUIT递达。 内核决定返回用户态后不是恢复main函数的上下文继续执行,而是执行sighandler函 数,sighandler比和main函数使用不同的堆栈空间,它们之间不存在调用和被调用的关系,是 两个独立的控制流程。 sighandler函数返
回后自动执行特殊的系统调用sigreturn再次进入内核态。 如果没有新的信号要递达,这次再返回用户态就是恢复

调用与屏蔽

当某个信号的处理函数被调用时,**内核自动将该信号加入进程的信号屏蔽字,**即处理期间,不允许再次调用。
当信号处理函数返回时自动恢复原来的信号屏蔽字,这样就保证了在处理某个信号时,如果这种信号再次产生,那么 它会被阻塞到当前处理结束为止。

结语

进程间通信到这里就暂时结束了(虽然内存池还差代码实现、共享内存还完全没写,但我准备之后用到了再写),希望对大家有帮助

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

jQuery-Word-Export 使用记录及完整修正文件下载 jquery.wordexport.js

参考资料&#xff1a; jQuery-Word-Export导出word_jquery.wordexport.js下载-CSDN博客 近期又需要自己做个 Html2Doc 的解决方案&#xff0c;因为客户又不想要 Html2pdf 的下载了&#xff0c;当初还给我费尽心思解决Html转pdf时中文输出的问题&#xff08;html转pdf文件下载之…

docker镜像、容器、仓库介绍

docker docker介绍docker镜像命令docker容器命令docker仓库 docker介绍 官网 Docker 是一种开源的容器化平台&#xff0c;用于开发、部署和运行应用。它通过将应用程序及其依赖项打包到称为“容器”的单一包中&#xff0c;使得应用能够在任何环境下运行&#xff0c;不受底层系…

51单片机-独立按键与数码管联动

独立键盘和矩阵键盘检测原理及实现 键盘的分类&#xff1a;编码键盘和非编码键盘 键盘上闭合键的识别由专用的硬件编码器实现&#xff0c;并产生键编码号或键值的称为编码键盘&#xff0c;如&#xff1a;计算机键盘。靠软件编程识别的称为非编码键盘&#xff1b;在单片机组成…

嵌入式驱动开发详解3(pinctrl和gpio子系统)

文章目录 前言pinctrl子系统pin引脚配置pinctrl驱动详解 gpio子系统gpio属性配置gpio子系统驱动gpio子系统API函数与gpio子系统相关的of函数 pinctrl和gpio子系统的使用设备树配置驱动层部分用户层部分 前言 如果不用pinctrl和gpio子系统的话&#xff0c;我们开发驱动时需要先…

STM32C011开发(1)----开发板测试

STM32C011开发----1.开发板测试 概述硬件准备视频教学样品申请源码下载参考程序生成STM32CUBEMX串口配置LED配置堆栈设置串口重定向主循环演示 概述 STM32C011F4P6-TSSOP20 评估套件可以使用户能够无缝评估 STM32C0 系列TSSOP20 封装的微控制器功能&#xff0c;基于 ARM Corte…

【漏洞复现】|百易云资产管理运营系统/mobilefront/c/2.php前台文件上传

漏洞描述 湖南众合百易信息技术有限公司&#xff08;简称&#xff1a;百易云&#xff09;成立于2017年是一家专注于不动产领域数字化研发及服务的国家高新技术企业&#xff0c;公司拥有不动产领域的数字化全面解决方案、覆盖住宅、写字楼、商业中心、专业市场、产业园区、公建、…

Spring Boot 实战:基于 Validation 注解实现分层数据校验与校验异常拦截器统一返回处理

1. 概述 本文介绍了在spring boot框架下&#xff0c;使用validation数据校验注解&#xff0c;针对不同请求链接的前端传参数据&#xff0c;进行分层视图对象的校验&#xff0c;并通过配置全局异常处理器捕获传参校验失败异常&#xff0c;自动返回校验出错的异常数据。 2. 依赖…

量子神经网络

感知机只是一个神经元&#xff0c;若有多个神经元共同作用&#xff0c;则构成神经网络。目前&#xff0c;最常见的量子神经网络模型为基于参数化量子线路的量子神经网络&#xff0c;该模型用参数化量子线路代替神经网络结构&#xff0c;使用经典优化算法更新参数化量子线路的参…

非交换几何与黎曼ζ函数:数学中的一场革命性对话

非交换几何与黎曼ζ函数&#xff1a;数学中的一场革命性对话 非交换几何&#xff08;Noncommutative Geometry, NCG&#xff09;是数学的一个分支领域&#xff0c;它将经典的几何概念扩展到非交换代数的框架中。非交换代数是一种结合代数&#xff0c;其中乘积不是交换性的&…

VUE3项目 关于金额:分转化为元 ;元转化为分;

1.在components 文件夹下新建moneyHandle.ts 文件 2.ts文件中写如下代码&#xff08;保留两位小数&#xff09; //分转化为元 - 正则解决精度 export const regFenToYuan (fen:any) >{var num fen;numfen*0.01;num;var reg num.indexOf(.) >-1 ? /(\d{1,3})(?(?:…

从0开始学习Linux——搭建自己的专属Linux系统

温馨提示本篇文章字数超过5000字&#xff01; 往期目录&#xff1a; 1、从0开始学习Linux——Linux简介&安装https://blog.csdn.net/diamond_5446/article/details/141190487 上一个章节我们简单了解了Linux&#xff0c;并且安装好了虚拟机以及下载好了Centos镜像&#…

《硬件架构的艺术》笔记(五):低功耗设计

介绍 能量以热量形式消耗&#xff0c;温度升高芯片失效率也会增加&#xff0c;增加散热片或风扇会增加整体重量和成本&#xff0c;在SoC级别对功耗进行控制就可以减少甚至可能消除掉这些开支&#xff0c;产品也更小更便宜更可靠。本章描述了减少动态功耗和静态功耗的各种技术。…

浅谈网络 | 传输层之TCP协议

目录 TCP 包头格式TCP 的三次握手TCP 的四次挥手TCP 的可靠性与"靠谱"的哲学TCP流量控制TCP拥塞控制 上一章我们提到&#xff0c;UDP 就像我们小时候一样简单天真&#xff0c;它相信“网之初&#xff0c;性本善&#xff0c;不丢包&#xff0c;不乱序”&#xff0c;因…

华为无线AC+AP组网实际应用小结

之前公司都是使用的H3C的交换机、防火墙以及无线AC和AP的&#xff0c;最近优化下无线网络&#xff0c;说新的设备用华为的&#xff0c;然后我是直到要部署的当天才知道用华为设备的&#xff0c;就很无语了&#xff0c;一点准备没有&#xff0c;以下为这次的实际操作记录吧&…

七、SElinux

一、SElinux简介 SELinux是Security-Enhanced Linux的缩写&#xff0c;意思是安全强化的linuxSELinux 主要由美国国家安全局(NSA)开发&#xff0c;当初开发的目的是为了避免资源的误用传统的访问控制在我们开启权限后&#xff0c;系统进程可以直接访问当我们对权限设置不严谨时…

Paddle Inference部署推理(一)

一&#xff1a;Paddle Inference推理 简介 Paddle Inference 是飞桨的原生推理库&#xff0c;提供服务器端的高性能推理能力。由于 Paddle Inference 能力直接基于飞桨的训练算子&#xff0c;因此它支持飞桨训练出的所有模型的推理。 Paddle Inference 功能特性丰富&#xff…

【数据分享】2001-2023年我国30米分辨率冬小麦种植分布栅格数据(免费获取)

小麦、玉米、水稻等各类农作物的种植分布数据在农业、环境、国土等很多专业都经常用到&#xff01; 本次给大家分享的是我国2001-2023年逐年的30米分辨率冬小麦种植分布栅格数据&#xff01;数据格式为TIFF格式&#xff0c;数据坐标为GCS_WGS_1984。该数据包括我国11个省份的冬…

类和对象plus版

一.类的定义 1.1类定义的格式 图中class为关键字&#xff0c;Stack为类的名字&#xff0c;用{}框住类的主体&#xff0c;类定义完后&#xff1b;不能省略。 为了区分成员变量&#xff0c;一般习惯在成员变量前面或后面加一个特殊标识&#xff0c;_或者m_ 1.2访问限定符 c采用…

jquery-picture-cut 任意文件上传(CVE-2018-9208)

目录 1、漏洞描述 2、访问ip&#xff1a;port 3、一句话木马&#xff1a;exploit.php 4、上传一句话木马 5、中国蚁剑连接成功 6、拿到flag 1、漏洞描述 jQuery是一个快速、简洁的JavaScript框架&#xff0c;是继Prototype之后又一个优秀的JavaScript代码库&#xff08;框…

【Spring Boot】# 使用@Scheduled注解无法执行定时任务

1. 前言 在 Spring Boot中&#xff0c;使用Scheduled注解来定义定时任务时&#xff0c;定时任务不执行&#xff1b;或未在规定时间执行。 import org.springframework.scheduling.annotation.Scheduled; import org.springframework.stereotype.Component;Component public c…