【Linux】—— 信号的产生
本期,我们今天要将的是信号的第二个知识,即信号的产生。
目录
(一)通过终端按键产生信号
(二)调用系统函数向进程发信号
(三)由软件条件产生信号
(四)硬件异常产生信号
(五)小结
(一)通过终端按键产生信号
SIGINT的默认处理动作是终止进程,SIGQUIT的默认处理动作是终止进程并且Core Dump,现在我们来验证一下。
我们之前学习进程等待的时候,给大家介绍了以下这张图片,其中【core dump】没有讲,今天我将给大家解释这个词的含义。
【解释说明】
- 首先解释什么是Core Dump。Linux提供了一种能力,当一个进程要异常终止时,可以选择把进程的用户空间内存数据全部保存到磁盘上,文件名通常是core,这叫做Core Dump(在云服务器上默认是关闭这个功能的!!!);
- 进程异常终止通常是因为有Bug,比如非法内存访问导致段错误,事后可以用调试器检查core文件以查清错误原因,这叫做Post-mortem Debug(事后调试);
- 一个进程允许产生多大的core文件取决于进程的Resource Limit(这个信息保存 在PCB中);
- 默认是不允许产生core文件的,因为core文件中可能包含用户密码等敏感信息,不安全;
- 在开发调试阶段可以用ulimit命令改变这个限制,允许产生core文件。 首先用ulimit命令改变Shell进程的Resource Limit,允许core文件最大为1024K: ulimit -c1024
命令用于显示有关用户当前资源限制(ulimits)的信息。此命令显示各种系统资源的硬限制和软限制,以下是一个示例输出:
【解释说明】
- core文件大小: core转储的最大大小(以块为单位)。
- 数据段大小: 进程的数据区大小(以千字节为单位)。
- 文件大小: 文件的最大大小(以块为单位)。
- 最大锁定内存: 锁定在内存中的地址空间的最大大小(以千字节为单位)。
- 最大内存大小: 进程可以拥有的最大数据大小(以千字节为单位)。
- 打开文件数: 进程可以拥有的最大文件描述符数量。
- 堆栈大小: 进程的堆栈大小(以千字节为单位)。
- CPU时间: 进程可以消耗的最大CPU时间(以秒为单位)。
- 最大用户进程数: 用户可以创建的最大进程数。
- 虚拟内存大小: 进程可用的虚拟内存大小(以千字节为单位)。
【注意】
- 值为“无限制”表示该特定资源没有设置特定限制;
- 可以使用 ulimit 命令或通过修改配置文件来调整这些限制。请注意,特定资源及其限制可能因操作系统和系统配置而异。
接下来,我们手动设置核心转储文件的大小。具体如下:
【解释说明】
- 上述命令用于设置核心文件的最大大小为10240个块(通常每个块大小为字节,具体大小取决于系统设置);
-c
: 表示设置或显示核心文件的最大大小限制。10240
: 表示核心文件的最大大小限制为10240个块。如果以字节计算,这意味着核心文件的最大大小为10240个块乘以每个块的字节数。
- 接下来,我们查询 signal信号。发现信号后面的有的是【term】,有的是【core】。具体如下:
【解释说明】
- 【term】 是一个信号名称,它代表着进程正常终止信号,无其他操作;
- 【core】通常指的是在进程异常终止(如段错误)时生成的核心转储文件。这个文件包含了进程在崩溃时的内存映像,可以用于调试和分析问题。当一个进程崩溃时,操作系统通常会生成一个名为
core
的文件,其中包含了进程在崩溃瞬间的内存状态。
接下来,我们就去验证上述结论:
- 首先,这里有几行简单的代码:
int main(int argc, char *argv[])
{while (true){cout << "我是一个正常运行的进程:" << getpid() << endl; sleep(1);}return 0;
}
- 紧接着,我们先让程序正常的跑起来:
- 程序可以正常运行之后,接下来,我先测试上述信号中后面标志为【term】的,看测试效果:
- 接下来,我先测试上述信号中后面标志为【core】的,看测试效果:
- 当我们打开这个文件时,发现全是乱码(因为这是给OS看的,不是给用户看的):
ulimit命令改变了Shell进程的Resource Limit,test进程的PCB由Shell进程复制而来,所以也具 有和Shell进程相同的Resource Limit值,这样就可以产生Core Dump了。 使用core文件:
(二)调用系统函数向进程发信号
首先在后台执行死循环程序,然后用kill命令给它发SIGSEGV信号
【解释说明】
- 3995是test进程的id。之所以要再次回车才显示 Segmentation fault ,是因为在3995进程终止掉 之前已经回到了Shell提示符等待用户输入下一条命令,Shell不希望Segmentation fault信息和用 户的输入交错在一起,所以等用户输入命令之后才显示。
- 指定发送某种信号的kill命令可以有多种写法,上面的命令还可以写成 kill -SIGSEGV 3995或 kill -11 3995, 11是信号SIGSEGV的编号。以往遇 到的段错误都是由非法内存访问产生的,而这个程序本身没错,给它发SIGSEGV也能产生段错误
kill命令是调用kill函数实现的。kill函数可以给一个指定的进程发送指定的信号。
- 函数的原型如下:
#include <signal.h>int kill(pid_t pid, int sig);
pid_t pid
: 要发送信号的进程的进程ID。如果pid
为正数,信号将发送到具有该进程ID的进程。如果pid
为0,信号将发送到与调用进程属于同一进程组的所有进程。如果pid
为-1,信号将发送到调用进程有权限发送信号的所有进程。int sig
: 要发送的信号的编号。
而raise函数可以给当前进程发送指定的信号(自己给自己发信号)
- 函数的原型如下:
#include <signal.h>int raise(int sig);
- int signo:要发送的信号的编号。
abort函数使当前进程接收到信号而异常终止
- 函数的原型如下:
#include <cstdlib>void abort(void);
使用abort函数非常简单,只需在需要终止进程的地方调用它即可。当调用abort函数时,以下操作将被执行:
- 向当前进程发送
SIGABRT
信号。 - 默认情况下,
SIGABRT
信号会导致进程终止,并生成一个核心转储文件。 - 终止处理程序会被启动,这是一个特殊的信号处理程序,可以用来执行一些清理工作或记录错误信息。
- 如果没有安装终止处理程序,或者终止处理程序调用了
_Exit
函数或返回,则进程会异常终止,并打印一条错误消息到标准错误流(stderr)。
代码展示:
void cleanup() {std::cout << "Performing cleanup before aborting..." << std::endl;// 执行一些清理工作
}void handler(int signo) {std::cout << "Received signal " << signo << std::endl;// 自定义信号处理逻辑exit(signo);
}int main()
{// 注册终止处理程序if (atexit(cleanup) != 0) {std::cerr << "Failed to register cleanup function" << std::endl;exit(EXIT_FAILURE);}// 注册信号处理函数if (signal(SIGABRT, handler) == SIG_ERR) {std::cerr << "Failed to register signal handler" << std::endl;exit(EXIT_FAILURE);}std::cout << "Starting program..." << std::endl;std::cout << "Triggering abort..." << std::endl;// 调用abort函数,触发进程终止abort();std::cout << "This line will not be reached" << std::endl;return 0;
}
输出展示:
【解释说明】
- 在调用abort函数后,程序收到了
SIGABRT
信号,并执行了注册的终止处理程序和信号处理函数; - 就像exit函数一样,abort函数总是会成功的,所以没有返回值
(三)由软件条件产生信号
SIGPIPE是一种由软件条件产生的信号,在“管道”中已经介绍过了。本节主要介绍alarm函数 和SIGALRM信号。
#include <unistd.h>
unsigned int alarm(unsigned int seconds);调用alarm函数可以设定一个闹钟,也就是告诉内核在seconds秒之后给当前进程发SIGALRM信号, 该信号的默认处理动作是终止当前进程。
- 接下来,我简单的演示一下这个函数:
//io的效率低下
int main()
{alarm(1);int count = 0;while (true) {// 打印,显示器打印网络std::cout << "count : " << count++ << std::endl; //1s之内计算机将一个正数累计到多少}return 0;
}
- 多跑几次程序我们可以发现打印出来的结果都是不同的:
- 当我们此时真正想测试计算机的算力时,我们可以像如下这样:
int count = 0;void myhandler(int signo)
{std::cout << "get a signal: " << signo << " count: " << count << std::endl;exit(0);
}int main(int argc, char *argv[])
{signal(SIGALRM,myhandler);alarm(1);while (true) count++;return 0;
}
- 多跑几次程序我们可以发现打印出来的结果跟上完全是天差地别:
- 上述 alarm 这个函数的返回值是0或者是以前设定的闹钟时间还余下的秒数;
- 打个比方,某人要小睡一觉,设定闹钟为30分钟之后响,20分钟后被人吵醒了,还想多睡一会儿,于是重新设定闹钟为15分钟之后响,“以前设定的闹钟时间还余下的时间”就是10分钟;
- 如果seconds值为0,表示取消以前设定的闹钟,函数的返回值仍然是以前设定的闹钟时间还余下的秒数
接下来,我们代码简单的演示一下:
void myhandler(int signo)
{std::cout << "get a signal: " << signo << " count: " << count << std::endl;int n = alarm(10);std::cout << "return: " << n << std::endl;
}int main(int argc, char *argv[])
{std::cout << "pid: " << getpid() << std::endl;signal(SIGALRM,myhandler);alarm(10);while(true){sleep(1);}return 0;
}
输出展示:
(四)硬件异常产生信号
硬件异常被硬件以某种方式被硬件检测到并通知内核,然后内核向当前进程发送适当的信号。例如当前进程执行了除以0的指令,CPU的运算单元会产生异常,内核将这个异常解释 为SIGFPE信号发送给进程。再比如当前进程访问了非法内存地址,,MMU会产生异常,内核将这个异常解释为SIGSEGV信号发送给进程。
(五)小结
以上便是本文的主要内容,接下来简单小结本文都讲了些什么!!!
在Linux中,信号可以通过多种方式产生,包括:
-
硬件异常: 这些是由硬件引起的异常事件,例如:
- 除零错误(SIGFPE): 当程序尝试执行除以零的操作时,会产生
SIGFPE
信号。 - 无效内存访问(SIGSEGV): 当程序尝试访问未分配给它的内存地址时,会产生
SIGSEGV
信号。 - 内存访问越界(SIGBUS): 当程序尝试访问无效的内存地址时,会产生
SIGBUS
信号。
- 除零错误(SIGFPE): 当程序尝试执行除以零的操作时,会产生
-
软件中断: 这些是由软件引发的事件,通常是为了通知进程已经达到了某个预定条件,例如:
- 定时器超时(SIGALRM): 当一个定时器达到设定的时间时,会产生
SIGALRM
信号。
- 定时器超时(SIGALRM): 当一个定时器达到设定的时间时,会产生
-
其他进程发送信号: 一个进程可以通过系统调用向另一个进程发送信号,例如:
- kill 命令: 通过命令行的
kill
命令或者在程序中使用kill
函数可以向指定的进程发送信号。 - 终端操作: 用户在终端中执行特定的操作,例如按下 Ctrl+C 组合键,会向当前前台进程发送
SIGINT
信号。
- kill 命令: 通过命令行的
-
进程自身发送信号: 进程可以通过调用
raise
函数或者kill
函数向自身发送信号。- 调用 kill 函数: 进程可以使用
kill
函数向自身发送信号,使用进程ID为getpid()
。
- 调用 kill 函数: 进程可以使用
-
软件条件满足: 在程序中,当特定的条件满足时,可以使用信号来通知其他部分程序执行某些动作,这通常需要由程序本身显示地触发。
以上便是本文的全部内容了,感谢大家的观看和支持!!!
相关文章:

【Linux】—— 信号的产生
本期,我们今天要将的是信号的第二个知识,即信号的产生。 目录 (一)通过终端按键产生信号 (二)调用系统函数向进程发信号 (三)由软件条件产生信号 (四)硬件…...
【算法】Hash 算法-关注优化细节
//给定一个未排序的整数数组 nums ,找出数字连续的最长序列(不要求序列元素在原数组中连续)的长度。 // // 请你设计并实现时间复杂度为 O(n) 的算法解决此问题。 // // // // 示例 1: // // //输入:nums [100,4…...

回归预测 | Matlab实现CPO-SVR冠豪猪优化支持向量机的数据多输入单输出回归预测
回归预测 | Matlab实现CPO-SVR冠豪猪优化支持向量机的数据多输入单输出回归预测 目录 回归预测 | Matlab实现CPO-SVR冠豪猪优化支持向量机的数据多输入单输出回归预测预测效果基本描述程序设计参考资料 预测效果 基本描述 1.Matlab实现CPO-SVR冠豪猪优化支持向量机的数据多输入…...

Idea设置代理后无法clone git项目
背景 对于我们程序员来说,经常上github找项目、找资料是必不可少的,但是一些原因,我们访问的时候速度特别的慢,需要有个代理,才能正常的访问。 今天碰到个问题,使用idea工具 clone项目,速度特…...
tkMapper 通用mapper的批量更新 批量新增 官方实现 springboot项目 依赖引入
文章目录 场景官方插件源码解析项目细节小结 场景 在许多业务场景下,需要对tkMapper的功能进行增强,需要用到批量新增和批量更新(这里是唯一主键去更新的),许多论文博客自己写的看起来并不行,我们这里就采…...
【leetcode刷刷】回溯:77.组合
77. 组合 第一次专门做回溯,有点难理解。首先可以理解回溯可以可视化为树的搜索,因此这道题,树的宽度为n,树的深度为kpath作为一个参数传入有点难想回溯没有返回值剪纸更难想,通过列算式可以勉强得到for的表达式&…...
【OOP】Python的OOP编程笔记
1.类变量和实例变量 类变量:变量属于类,在对象中是共用的。访问方式为类名.变量名,或对象名.__class__.变量名 实例变量:定义在方法中的变量,属于具体对象。访问方式为对象名.变量名 类变量访问方式 class Car:# nu…...

一进一出模拟量信号隔离变送器
一进一出模拟量信号隔离变送器 捷晟达科技推出一进一出模拟量信号隔离变送器 深圳捷晟达科技推出一款具有隔离,放大,转换保护功能的一进一出的小型隔离变送器设备,该设备可以把模拟量(4-20mA/0-10V等)标准信号转换用户需要的信号,该产品具有抗EMC干扰,可以有效的保护后级设备安…...

Mybatis-plus原生pages分页未生效的解决方案
文章目录 前言原因1、Mybatis Plus版本的问题2、Mapper.xml文件中SQL语句格式问题3、Mybatis Plus默认分页拦截器问题4、分页参数传参问题5、分页配置的问题 解决方案1、升级对应的Mybatis-plus版本分页插件配置问题3、自定义分页拦截器4、正确的参数5、不同版本的配置文件3.4.…...

【linux】-centos7版本前后-变化篇
1.centos7版本前后区别 首先文件系统变化,由EXT4,变为XFS格式。可支持容量500TB的文件,而6代仅能支持16TB。首个进程变为systemd, 替换了熟悉的init进程。它的特点是功能强大,体积也很强大。 systemd给我们带来了一个全家桶命令&…...

001集—shapefile(.shp)格式详解——arcgis
一、什么是shapefile Shapefile 是一种用于存储地理要素的几何位置和属性信息的非拓扑简单格式。shapefile 中的地理要素可通过点、线或面(区域)来表示。包含 shapefile 的工作空间还可以包含 dBASE 表,它们用于存储可连接到 shapefile 的要…...
ssrf服务器请求伪造漏洞(个人学习)
SSRF前置学习须了解net工作原理 计算机网络 网络地址转换NAT_内部本地地址-CSDN博客 可以看这个来了解 SSRF 攻击的目标:从外网无法访问的内部网络 形成原因:大部分服务器提供了从外部应用获取数据的功能,但是对目标地址没有做过滤和限制…...

【前端web入门第二天】03 表单-下拉菜单 文本域 label标签 按钮 【附注册信息综合案例】
文章目录: 1. 下拉菜单 2. 文本域3.label标签 4.按钮- button 4.1 reset重置按钮结合form表单区域使用 5.无语义的布局标签 6.字符实体 注册信息综合案例 表单第二节 1. 下拉菜单 标签: select嵌套option,select是下拉菜单整体,option是下拉菜单的每一项。 代码…...
回响科技二面面试题解答
面试题 1、你们的数仓中DWD层为什么要划分数据域?划分数据域之后会对ADS层造成什么影响?是可以提效还是可扩展性强?你们是如何考虑的呢? 2、AZkaban和dolphinScheduler的区别是什么?如果选型会从哪几个方面来考虑呢&a…...

node学习过程中的终端命令
冷的哥们手真tm冷,打字都是僵的,屮 目录 一、在学习nodejs过程中用到的终端命令总结 一、在学习nodejs过程中用到的终端命令 node -v nvm install 20.11.0 nvm list nvm list available nvm on nvm -v nvm use 20.11.0 node加要运行的js文件路径 ps&a…...
oracle版本号中的i,G,C代表什么含义
大家都熟悉的 Oracle 版本号有 9i、10G、11G、12C、19C 等,但在早期,Oracle 的版本号并不包含这些字母。 最初,Oracle 的版本号简单地是 1、2、3、4 等,一直发展到 1999 年发布的 8i 版本。20 世纪末是互联网爆发式发展的时代。 …...
Unity2D_角色移动跳跃
水平移动 Rigidbody2D 使用Unity自带的刚体组件,通过修改刚体物品中Rigidbody2D的属性velocity速度来实现物体移动 声明Rigidbody2D属性,以及角色移动速度 public float playerSpeed 5f; private Rigidbody2D rigidbody2D;在Start方法中将rigidbody2D…...
23-GPTs Actions详细解析:如何查找、对接API,及如何编写Schema
目录 1. 挖掘一个API接口,并将其对接到GPTs中1.1 如何寻找API1.2 把API对接到Schema中1.3 Schema中的参数结构本节详细介绍GPTs Actions 中的Schema。如何写、它里面的结构是什么、里面参数的含义。 后续使用GPTs过程中,如何找到自己想要的API,以及如何把找到的API对接到GPT…...
微信小程序canvas画布实现椭圆元素自由缩放、移动功能
目录 实现效果 编辑 一、获取画布信息并绘制背景 二、绘制椭圆...
使用Excel计算--任务完成总工作日时长
(Owed by: 春夜喜雨 http://blog.csdn.net/chunyexiyu) 引言 计算任务完成时间周期,和计算金钱一样,是一个比较细致严谨的工作。 通常,我们可能以为,完成周期形如: 任务完成周期 任务结束时间 - 任务开始时间 但是…...

网络六边形受到攻击
大家读完觉得有帮助记得关注和点赞!!! 抽象 现代智能交通系统 (ITS) 的一个关键要求是能够以安全、可靠和匿名的方式从互联车辆和移动设备收集地理参考数据。Nexagon 协议建立在 IETF 定位器/ID 分离协议 (…...

基于距离变化能量开销动态调整的WSN低功耗拓扑控制开销算法matlab仿真
目录 1.程序功能描述 2.测试软件版本以及运行结果展示 3.核心程序 4.算法仿真参数 5.算法理论概述 6.参考文献 7.完整程序 1.程序功能描述 通过动态调整节点通信的能量开销,平衡网络负载,延长WSN生命周期。具体通过建立基于距离的能量消耗模型&am…...
模型参数、模型存储精度、参数与显存
模型参数量衡量单位 M:百万(Million) B:十亿(Billion) 1 B 1000 M 1B 1000M 1B1000M 参数存储精度 模型参数是固定的,但是一个参数所表示多少字节不一定,需要看这个参数以什么…...
三维GIS开发cesium智慧地铁教程(5)Cesium相机控制
一、环境搭建 <script src"../cesium1.99/Build/Cesium/Cesium.js"></script> <link rel"stylesheet" href"../cesium1.99/Build/Cesium/Widgets/widgets.css"> 关键配置点: 路径验证:确保相对路径.…...
深入浅出:JavaScript 中的 `window.crypto.getRandomValues()` 方法
深入浅出:JavaScript 中的 window.crypto.getRandomValues() 方法 在现代 Web 开发中,随机数的生成看似简单,却隐藏着许多玄机。无论是生成密码、加密密钥,还是创建安全令牌,随机数的质量直接关系到系统的安全性。Jav…...
Auto-Coder使用GPT-4o完成:在用TabPFN这个模型构建一个预测未来3天涨跌的分类任务
通过akshare库,获取股票数据,并生成TabPFN这个模型 可以识别、处理的格式,写一个完整的预处理示例,并构建一个预测未来 3 天股价涨跌的分类任务 用TabPFN这个模型构建一个预测未来 3 天股价涨跌的分类任务,进行预测并输…...
在 Nginx Stream 层“改写”MQTT ngx_stream_mqtt_filter_module
1、为什么要修改 CONNECT 报文? 多租户隔离:自动为接入设备追加租户前缀,后端按 ClientID 拆分队列。零代码鉴权:将入站用户名替换为 OAuth Access-Token,后端 Broker 统一校验。灰度发布:根据 IP/地理位写…...

Razor编程中@Html的方法使用大全
文章目录 1. 基础HTML辅助方法1.1 Html.ActionLink()1.2 Html.RouteLink()1.3 Html.Display() / Html.DisplayFor()1.4 Html.Editor() / Html.EditorFor()1.5 Html.Label() / Html.LabelFor()1.6 Html.TextBox() / Html.TextBoxFor() 2. 表单相关辅助方法2.1 Html.BeginForm() …...

打手机检测算法AI智能分析网关V4守护公共/工业/医疗等多场景安全应用
一、方案背景 在现代生产与生活场景中,如工厂高危作业区、医院手术室、公共场景等,人员违规打手机的行为潜藏着巨大风险。传统依靠人工巡查的监管方式,存在效率低、覆盖面不足、判断主观性强等问题,难以满足对人员打手机行为精…...
【学习笔记】erase 删除顺序迭代器后迭代器失效的解决方案
目录 使用 erase 返回值继续迭代使用索引进行遍历 我们知道类似 vector 的顺序迭代器被删除后,迭代器会失效,因为顺序迭代器在内存中是连续存储的,元素删除后,后续元素会前移。 但一些场景中,我们又需要在执行删除操作…...