Linux:信号的概念与产生
信号概念
信号是进程之间事件异步通知的一种方式
在Linux命令行中,我们可以通过ctrl + c
来终止一个前台运行的进程,其实这就是一个发送信号的行为。我们按下ctrl + c
是在shell
进程中,而被终止的进程,是在前台运行的另外一个进程。因此信号是一种进程之间的通知方式。
可以通过指令kill -l
来查询信号:以上就是Linux中的全部信号,它们分为两个区间:[1, 31]
和[34, 64]
,也就是说没有0,32,33
这三个信号,虽然信号的最大编号为64
,但实际上只有62
个信号。
[1, 31]
:这些信号称为非实时信号
,当进程收到这些信号后,可以自己选择合适的时候处理[34, 64]
:这些信号称为实时信号
,当进程收到这些信号后,必须立马处理
由于现在的操作系统基本都是分时操作系统,因此实时信号其实是不符合设计理念的,几乎用不到实时信号,本博客只讲解非实时信号
。
所有信号都是大写的单词,在C/C++
中,一般来说宏
就是大写的,其实信号名就是宏。
那么进程收到信号后要怎么处理呢?
进程有三种处理信号的方式:
- 忽略此信号
- 执行信号的默认处理函数
- 执行信号的自定义处理函数,这种方式也称为
信号捕捉
其中Term
,Ign
,Core
,Stop
,Cont
就是信号处理的默认行为
Term
:默认操作是终止进程Ign
:默认操作是忽略信号Core
:默认操作是终止进程并转储核心Stop
:默认操作是暂停进程Cont
:默认操作是,如果该进程当前已暂停,则继续该进程
signal
函数,包含在头文件<signal.h>
中,可以自定义信号的处理方式,函数原型如下:
sighandler_t signal(int signum, sighandler_t handler);
其中这个sighandler_t
类型,本质是一个void (*)(int)
类型的函数指针,也就是说自定义的信号处理函数必须是void (int)
的格式。其中这个处理函数的第一个参数int
,就是用来接收信号的编号的。
而返回值也是sighandler_t
,其返回原先该信号处理的函数的函数指针。
void handler(int sig)
{cout << "get sig: " << sig << endl;
}int main()
{signal(2, handler);while(true){cout << "hello world!" << endl;sleep(1);}return 0;
}
以上代码中,我们通过signal(2, handler);
把2
号信号的处理方式变成了执行函数handler
,此后进程收到2
号信号时,就会执行cout << "get sig: " << sig << endl;
了。
而2
号信号SIGINT
就是ctrl + C
发送的信号,因此我们可以直接在shell
中通过ctrl + C
来发送2
号信号,从而验证效果。
信号产生
简单讲解了一下信号的三种处理方式后,再来看看信号是如何产生的,在Linux中,信号主要有两种产生方式:软件条件
,硬件异常
。
软件信号
在 Linux 中,“软件条件” 发出的信号指的是由 进程自身或其他进程 产生的信号,而不是由硬件中断或其他外部事件触发的信号。
Linux中有多种系统调用可以发送信号,在此我讲解kill
,raise
,abort
,alarm
四种接口,其中abort
并不是一个系统调用,而是一个用户操作接口。
kill
kill
函数用于给指定pid
的进程发送指定信号,需要头文件<sys/types.h>
和<signal.h>
,函数原型如下:
int kill(pid_t pid, int sig);
参数:
pid
:收到该信号的进程的pid
sig
:发送哪一个信号
返回值:
- 返回
0
:发送信号成功 - 返回
-1
:发送信号失败
void handler (int sig)
{cout << "get sig: " << sig << endl;exit(1);
}int main()
{pid_t id = fork();if (id == 0) // 子进程{signal(2, handler);while (true){cout << "I am child process" << endl;sleep(1);}}sleep(5);kill(id, 2);return 0;
}
以上示例中,父进程通过fork创建子进程,sleep五秒后通过kill(id, 2);给子进程发送(2) SIGINT信号。子进程通过signal(2, handler);修改了信号处理方式,随后每秒钟输出一次I am child process。
在handler中,会先输出get sig: 2,表示自己收到了信号,然后exit退出进程。
另外的,也可以通过kill
指令发送信号,格式为:
kill -sig pid
其中sig
为要发送的信号,pid
为收到信号的进程pid
。
其实kill
指令底层就是调用kill
接口,依然属于系统调用的范围。
raise
raise
函数用于给自己发送信号,需要头文件<signal.h>
,函数原型如下:
int raise(int sig);
参数:
sig
:发送哪一个信号
返回值:
- 返回
0
:发送信号成功 - 返回
-1
:发送信号失败
void handler(int sig)
{cout << "get sig: " << sig << endl;exit(1);
}int main()
{signal(2, handler);int cnt = 5;while (cnt){cout << "I am a process cnt = " << cnt-- << endl;sleep(1);}raise(2);return 0;
}
先通过signal(2, handler);
修改信号的处理函数,随后循环五次,输出"I am a process cnt = "
,最后通过raise(2);
给自己发送(2) SIGINT
信号。
abort
abort
函数用于给自己发送(6) SIGABRT
信号,需要头文件<stdlib.h>
,属于用户操作接口
,函数原型如下:
void abort(void);
这个函数功能十分简单,就是给自己发送(6) SIGABRT
信号,示例:
int main()
{abort();return 0;
}
输出结果:
Aborted
最后进程输出了Aborted
表示自己收到了(6) SIGABRT
信号。
alarm
alarm
函数用于设定一个闹钟,在一定之间后给当前进程发送信号(14) SIGALRM
,需要头文件<unistd.h>
,函数原型如下:
unsigned int alarm(unsigned int seconds);
参数:
seconds
:在seconds
秒后发送信号
返回值:
- 如果之前有还没响的闹钟:取消上一次的闹钟,并返回上一次闹钟的剩余秒数
- 如果之前没有闹钟了:返回
0
void handler(int sig)
{cout << "get sig: " << sig << endl;exit(1);
}int main()
{signal(SIGALRM, handler);alarm(1);int i = 0;while(true){cout << i++ << endl;}return 0;
}
以上代码中,先通过signal(SIGALRM, handler);
自定义信号(14) SIGALRM
的处理方式。然后通过alarm(1);
设定一秒钟的闹钟,在一秒内,程序会不断执行while
循环让i++
,我们可以看看一秒内计算机可以执行多少次i++
。
输出结果:
74560
get sig:14
可以看到,计算到74560
时,就收到了SIGALRM
,终止进程了。
硬件信号
硬件信号指的是由 硬件事件 触发的信号,而不是由软件代码逻辑控制的。
具体来说,以下情况属于硬件条件发出的信号:
- 中断: 硬件设备,例如键盘、鼠标、网络接口等,在发生事件时会向 CPU 发送中断信号,例如键盘按键按下、网络数据包到达等。
- 异常: CPU 在执行指令过程中,如果遇到错误情况,例如除以零、内存访问错误等,会产生异常信号。
- 时钟中断: 系统定时器会定期向 CPU 发送时钟中断信号,用于调度进程和执行定时任务。
键盘产生信号
通过键盘发送信号是最简单的信号发送方式,最常用的有ctrl + C
和ctrl + \
。
ctrl + C
:向前台进程发送(2) SIGINT
信号,效果为直接终止进程ctrl + \
:向前台进程发送(3) SIGQUIT
信号,效果为直接终止进程
硬件中断
硬件给进程发送信号的本质,其实是通过硬件中断
。
硬件中断是指硬件设备向 CPU 发送的信号,通知 CPU 有事件发生需要处理。当一个硬件设备需要向 CPU 汇报事件发生时,它会向 CPU 发送一个中断信号。CPU 接收这个信号后,会暂停当前执行的任务,转而去处理硬件设备的请求。
int main()
{int a = 5 / 0;return 0;
}
这是一个很经典的除零错误,如果我们强行运行这个进程,会报出错误Floating point exception
,本质上是收到了信号(8) SIGFPE
,其实本质上是发生了硬件中断。
CPU
:CPU
是一个硬件,其要处理计算,而5/0
这个计算过程就是在CPU
中完成的,当CPU
检测到这个计算中0
做了除数,于是对自己发起硬件中断
操作系统
:操作系统检测到硬件中断
后,跳转到中断处理程序
(中断处理程序
是Linux
内核的一部分),然后检测该硬件中断是什么原因,发现是因为除零错误
,于是给进程发送(8) SIGPFE
信号进程
:进程收到操作系统发来的(8) SIGPFE
信号后,执行相应的处理措施
我们之前的ctrl + C
按键发送(2) SIGINT
信号,本质也是硬件中断,当我们从键盘输入了数据后,键盘向CPU
发出硬件中断,随后CPU
去执行操作系统
中的硬件中断程序
,发现是用户按下了ctrl + C
,于是操作系统向进程发送(2) SIGINT
信号。
与软件条件信号相比,硬件条件信号具有以下特点:
- 由硬件触发: 硬件条件信号的产生是由硬件事件触发的,而不是由代码逻辑决定的
- 不可控性: 硬件条件信号的产生通常无法被进程控制,例如硬件设备的故障、网络连接中断等。
- 不可预测性: 硬件条件信号的产生通常是不可预测的,因为它们是由硬件事件触发的。
相关文章:
Linux:信号的概念与产生
信号概念 信号是进程之间事件异步通知的一种方式 在Linux命令行中,我们可以通过ctrl c来终止一个前台运行的进程,其实这就是一个发送信号的行为。我们按下ctrl c是在shell进程中,而被终止的进程,是在前台运行的另外一个进程。因…...
云监控(华为) | 实训学习day2(10)
spring boot基于框架的实现 简单应用 - 用户数据显示 开发步骤 第一步:文件-----》新建---项目 第二步:弹出的对话框中,左侧选择maven,右侧不选任何内容. 第三步,选择maven后,下一步 第4步 :出现对话框中填写项目名称 第5步&…...
数据结构第35节 性能优化 算法的选择
算法的选择对于优化程序性能至关重要。不同的算法在时间复杂度、空间复杂度以及适用场景上有着明显的差异。下面我将结合具体的代码示例,来讲解几种常见的算法选择及其优化方法。 示例 1: 排序算法 场景描述: 假设我们需要对一个整数数组进行排序。 算法选择: …...
每天一个数据分析题(四百三十六)- 正态分布
X为服从正态分布的随机变量N(2, 9), 如果P(X>c)P(X<c), 则c的值为() A. 3 B. 2 C. 9 D. 2/3 数据分析认证考试介绍:点击进入 题目来源于CDA模拟题库 点击此处获取答案 数据分析专项练习题库 内容涵盖Python,SQL&…...
跟我学C++中级篇——虚函数的性能
一、虚函数性能 一般来说,面向对象的设计中,继承和多态是其中两个非常重要的特征。从使用的过程来看,一般应用到继承的,使用多态的可能性就非常大。而多态的实现有很多种, 但开发者通常认为的多态(动多态&…...
trl - 微调、对齐大模型的全栈工具
文章目录 一、关于 TRL亮点 二、安装1、Python包2、从源码安装3、存储库 三、命令行界面(CLI)四、如何使用1、SFTTrainer2、RewardTrainer3、PPOTrainer4、DPOTrainer 五、其它开发 & 贡献参考文献最近策略优化 PPO直接偏好优化 DPO 一、关于 TRL T…...
GuLi商城-商品服务-API-品牌管理-品牌分类关联与级联更新
先配置mybatis分页: 品牌管理增加模糊查询: 品牌管理关联分类: 一个品牌可以有多个分类 一个分类也可以有多个品牌 多对多的关系,用中间表 涉及的类: 方法都比较简单,就不贴代码了...
【linux】服务器ubuntu安装cuda11.0、cuDNN教程,简单易懂,包教包会
【linux】服务器ubuntu安装cuda11.0、cuDNN教程,简单易懂,包教包会 【创作不易,求点赞关注收藏】 文章目录 【linux】服务器ubuntu安装cuda11.0、cuDNN教程,简单易懂,包教包会一、版本情况介绍二、安装cuda1、到官网…...
在 Apifox 中如何高效批量添加接口请求 Body 参数?
在使用 Apifox 进行 API 设计时,你可能会遇到需要添加大量请求参数的情况。想象一下,如果一个接口需要几十甚至上百个参数,若要在接口的「修改文档」里一个个手动添加这些参数,那未免也太麻烦了,耗时且易出错。这时候&…...
专业PDF编辑工具:Acrobat Pro DC 2024.002.20933绿色版,提升你的工作效率!
软件介绍 Adobe Acrobat Pro DC 2024绿色便携版是一款功能强大的PDF编辑和转换软件,由Adobe公司推出。它是Acrobat XI系列的后续产品,提供了全新的用户界面和增强功能。用户可以借助这款软件将纸质文件转换为可编辑的电子文件,便于传输、签署…...
车载音视频App框架设计
简介 统一播放器提供媒体播放一致性的交互和视觉体验,减少各个媒体应用和场景独自开发的重复工作量,实现媒体播放链路的一致性,减少碎片化的Bug。本文面向应用开发者介绍如何快速接入媒体播放器。 主要功能: 新设计的统一播放U…...
StarRocks on AWS Graviton3,实现 50% 以上性价比提升
在数据时代,企业拥有前所未有的大量数据资产,但如何从海量数据中发掘价值成为挑战。数据分析凭借强大的分析能力,可从不同维度挖掘数据中蕴含的见解和规律,为企业战略决策提供依据。数据分析在营销、风险管控、产品优化等领域发挥…...
VUE中setup()
在Vue中,setup() 函数是Vue 3.0及更高版本引入的一个重要特性,它是Composition API的入口点。setup() 函数用于初始化组件的状态和逻辑,包括定义响应式数据、方法和生命周期钩子。以下是关于setup() 函数的详细解释: 1. 作用与特…...
【单元测试】SpringBoot
【单元测试】SpringBoot 1. 为什么单元测试很重要?‼️ 从前,有一个名叫小明的程序员,他非常聪明,但有一个致命的缺点:懒惰。小明的代码写得又快又好,但他总觉得单元测试是一件麻烦事,觉得代码…...
分布式搜索引擎ES-elasticsearch入门
1.分布式搜索引擎:luceneVS Solr VS Elasticsearch 什么是分布式搜索引擎 搜索引擎:数据源:数据库或者爬虫资源 分布式存储与搜索:多个节点组成的服务,提高扩展性(扩展成集群) 使用搜索引擎为搜索提供服务。可以从海量…...
TCP三次握手与四次挥手详解
1.什么是TCP TCP(Transmission Control Protocol,传输控制协议)是一种面向连接的、可靠的、基于字节流的通信协议,属于互联网协议族(TCP/IP)的一部分。TCP 提供可靠的、顺序的、无差错的数据传输服务&…...
【Windows】操作系统之任务管理器(第一篇)
一、操作系统简介 Windows操作系统是由微软公司(Microsoft)开发的一款图形操作系统,它以其强大的功能和广泛的用户基础,成为了目前世界上用户使用最多、兼容性最强的操作系统之一。以下是关于Windows操作系统的详细介绍ÿ…...
图同构的必要条件
来源:离散数学...
Django获取request请求中的参数
支持 post put json_str request.body # 属性获取最原始的请求体数据 json_dict json.loads(json_str)# 将原始数据转成字典格式 json_dict.get("key", "默认值") # 获取数据参考 https://blog.csdn.net/user_san/article/details/109654028...
kotlin compose 实现应用内多语言切换(不重新打开App)
1. 示例图 2.具体实现 如何实现上述示例,且不需要重新打开App ①自定义 MainApplication 实现 Application ,定义两个变量: class MainApplication : Application() { object GlobalDpData { var language: String = "" var defaultLanguage: Strin…...
记录些MySQL题集(16)
MySQL 存储过程与触发器 一、初识MySQL的存储过程 Stored Procedure存储过程是数据库系统中一个十分重要的功能,使用存储过程可以大幅度缩短大SQL的响应时间,同时也可以提高数据库编程的灵活性。 存储过程是一组为了完成特定功能的SQL语句集合&#x…...
【算法基础】Dijkstra 算法
定义: g [ i ] [ j ] g[i][j] g[i][j] 表示 v i v_i vi 到 $v_j $的边权重,如果没有连接,则 g [ i ] [ j ] ∞ g[i][j] \infty g[i][j]∞ d i s [ i ] dis[i] dis[i] 表示 v k v_k vk 到节点 v i v_i vi 的最短长度, …...
使用 Flask 3 搭建问答平台(三):注册页面模板渲染
前言 前端文件下载 链接https://pan.baidu.com/s/1Ju5hhhhy5pcUMM7VS3S5YA?pwd6666%C2%A0 知识点 1. 在路由中渲染前端页面 2. 使用 JinJa 2 模板实现前端代码复用 一、auth.py from flask import render_templatebp.route(/register, methods[GET]) def register():re…...
pycharm如何debug for循环里面的错误值
一般debug时,在for循环里面的话,需要自己一步一步点。如果循环几百次那种就比较麻烦。此时可以采用try except的方式来解决 例子如下 #ptyhon debug for循环的代码 num[1,2,3,s,4] ans0 for i in num:try:ansiexcept:print(错误) print(ans) 结果如下&a…...
解决网页中的 video 标签在移动端浏览器(如百度访问网页)视频脱离文档流播放问题
问题现象 部分浏览器视频脱离文档流,滚动时,视频是悬浮出来,在顶部播放 解决方案 添加下列属性,可解决大部分浏览器的脱离文档流的问题 <videowebkit-playsinline""playsInlinex5-playsinlinet7-video-player-t…...
.Net--CLS,CTS,CLI,BCL,FCL
1.什么是CLS? 所以.NET专门为此参考每种语言(例如C# ,VB,F#)并找出了语言间的共性,然后定义了一组规则,开发者都遵守这个规则来编码,那么代码就能被任意.NET平台支持的语言所通用。 而与其说是规则&#x…...
Stable Diffusion:质量高画风清新细节丰富的二次元大模型二次元插图
今天和大家分享一个基于Pony模型训练的二次元模型:二次元插图。关于该模型有4个不同的分支版本。 1.5版本:loar模型,推荐底模型niji-动漫二次元4.5。 xl版本:SDXL模型版本 mix版本:光影减弱,减少SDXL版本…...
数读MEME之争:以太坊获更高价值共识,抢占热点成Solana流量密码
在当前显著的加密牛市中,以太坊和Solana之间的竞争不仅在币价表现上显而易见,生态发展方面也备受关注。特别是在这轮MEME行情中,双方阵营的MEME代币呈现出不同的特点和趋势。 市场表现对比 以太坊的优势: 市场份额和认可度更高&…...
python的with语句
1.with语句的作用 在 Python 中,with 语句用于创建一个上下文管理器,以更简洁和安全的方式管理资源。 其主要优点是可以确保在代码块执行完毕后,相关资源能够被正确释放或清理,即使在代码块内部发生了异常。 以下是一个使用 with…...
Selenium原理深度解析
在自动化测试领域,Selenium无疑是最受欢迎和广泛使用的工具之一。它支持多种浏览器和操作系统,为开发人员和测试人员提供了强大的自动化测试解决方案。本文将深入探讨Selenium的工作原理,包括其架构、核心组件、执行流程以及它在自动化测试中…...
附近做网站/网络推广项目外包公司
神经网络的结构设计有3个主流的高级技巧:1,高低融合 (将高层次特征与低层次特征融合,提升特征维度的丰富性和多样性,像人一样同时考虑整体和细节)2,权值共享 (一个权值矩阵参与多个不同的计算,降低参数规模…...
网站建设续费是什么费用/轻松seo优化排名
今天上午的蓝桥杯比赛让我正视了很多以前没有注意的问题,也找到了很多自以为会了,其实是盲点的知识点。 10个题都不难,放到平时,如果是在oj上,肯定能全部AC,可是比赛的时候,写的并不好。 尤其…...
做英文网站2014/运营推广计划
2019独角兽企业重金招聘Python工程师标准>>> http://elixir-lang.github.io/getting-started/binaries-strings-and-char-lists.html https://elixirforum.com/t/bitstring-and-binary/2351/4 字符串是怎么保存到计算机中的 iex>?a 97 iex>?吴 21556 a的cod…...
免费注册自己的网站/培训课
1354 选数字基准时间限制:1 秒 空间限制:131072 KB 分值: 80 难度:5级算法题收藏关注当给定一个序列a[0],a[1],a[2],...,a[n-1] 和一个整数K时,我们想找出,有多少子序列满足这么一个条件:把当前子序列里面的所有元素乘…...
wordpress皮肤下载/博客网站注册
一、简介 Oracle权限分为系统权限和对象权限。 1、系统权限 注意:系统权限不支持级联回收,所以你需要使用sysdba一个个的回收。 2、对象权限 注:对象权限支持级联回收,系统权限不支持级联回收 1、查询oracle中的所有的权限,必须是sysdba才能进行查询 select * from system_priv…...
互联业务登录页 网站/百度网址大全设为主页
空姐梅梅入住酒店,意外发现房间内装有针孔摄像头。梅梅认为自己的个人隐私被严重侵犯,要求酒店担责,但酒店却称并不知情,而且摄像头早已陈旧损坏,并没有实际摄录功能。近日,法院经审理认定酒店方侵权&#…...