【Linux】 进程信号的发生

在这里插入图片描述

送给大家一句话:
何必向不值得的人证明什么,生活得更好,乃是为你自己。
-- 亦舒

进程信号的发生

  • 1 何为信号
  • 2 信号概念的基础储备
  • 3 信号产生
    • kill系统调用
    • alarm系统调用
    • 异常
    • core term
  • Thanks♪(・ω・)ノ谢谢阅读!!!
  • 下一篇文章见!!!

1 何为信号

生活中信号随处可见,我们的生活离不开信号。就比如过红路灯,看见绿灯亮的信号,我们就可以过马路了;听见闹钟响了,我们知道时间到了;看见别人脸色不好,我们就知道他有心事…所以信号在生活中随时可以产生(与我们的动作是异步的):

  1. 我们可能认识这个信号,知道这个信号如何处理。
  2. 但是当信号产生时,我们可能在做着其他事情,会把信号暂时不做处理。
  3. 暂不处理就要求我们记得这个信号,并确定什么时候处理。

对此,如果把上面的“我们”换成“进程”,就是进程中的信号了!我们可以看看在Linux系统下的信号:
在这里插入图片描述
信号时从 1 - 64 的数字对应信号(32 - 64 是实时信号,暂不考虑)

信号的生命周期可以划分为:预备 -> 产生 -> 保存 -> 处理 。我们把这个过程研究明白就可以了

2 信号概念的基础储备

信号是Linux系统通过的一种向目标进程发送指定事件的方式。要做识别和处理。

信号产生时异步的:对于一个进程不知道什么时候会收到信号,他只能先做自己的事情,信号产生时也不知道进程在干什么,所以是异步的!!!

先说明一个概念信号处理有三种(只能三选一):

  1. 默认动作 — 进程处理信号都是默认的 ,通常是终止自己(term , core),暂停 ,忽略…
    在这里插入图片描述

  2. 忽略动作

  3. 自定义动作 — 信号的捕捉 : signal(int signum, sighandler_t handler) ,该函数可以捕捉signum信号,并执行自定义的handler函数
    在这里插入图片描述

接下来我们来看看signal函数的使用:对信号的自定义捕捉 ,只需捕捉一次,后续就一直有效!!!

#include <signal.h>
#include <iostream>
#include <sys/types.h>
#include <unistd.h>

void handler(int sig)
{
    std::cout << "get sig:" << sig << std::endl;
}

int main()
{
	//对信号的自定义捕捉 ,只需捕捉一次,后续就一直有效!!!
    signal(2, handler);
    while (true)
    {
        std::cout << "pid :" << getpid() << std::endl;
        sleep(1);
    }
    return 0;
}

该程序的功能是每隔一秒打印出当前进程的ID,并且当接收到SIGINT信号*(2号信号)时,会调用handler函数打印出信号编号。通常情况下,当你运行这个程序并按下Ctrl+C时,程序会打印出"get sig:2"并继续运行,因为它已经处理了SIGINT信号。如果你想要终止程序,使用kill命令。

来看效果:
在这里插入图片描述
我们也可以多设置一些捕捉:

signal(2, handler);
signal(3, handler);
signal(4, handler);
signal(5, handler);

这样可以与之捕捉对应的信号!

如何理解信号的发送和处理??? 浅度理解:

进程我们知道是通过task_struct来管理的,里面有一个uint32_t signal变量,其收到的信号是通过位图来保存的(1 - 31比特位分别代表 1 - 31 号信号)。
那么发送信号:就是修改指定进程pcb的信号的指定位图 0 -> 1。也就是“写信号”
通过上面的分析,发送信号改变了内核数据结构,而这个工作只能是OS来进行,所以只有OS可以发送信号

那么kill信号能向进程发送信号!还有其他可以发送信号的方式吗?我们来看:

3 信号产生

信号产生的方式有以下几种:

  1. 通过kill命令:向指定进程发送指定的信号
  2. 键盘可以产生信号:我们常用的ctrl + c (2号信号)和 ctrl + \(3号信号)都可以向进程发送信号
  3. 系统调用:
    • int kill(pid_t pid, int sig)
      在这里插入图片描述
      该系统调用可以向pid对应的进程,发送sig信号。发送成功返回 1 反之返回 0。
    • 还有 int raise(int sig); 系统调用是向当前进程发送指定信号。比较简单奥。
    • 还有 void abort(void); 库函数异常终止当前进程。是对应的6号信号(终止会打印Aborted!)其特殊的性质是可以被捕捉,但是进程还是会被终止掉,就是为了防止发生所有信号都被捕捉,没有信号可以终止的情况,9号信号和19号信号不能被自定义捕捉!!!
  4. 软件条件:
    我们回忆一下:管道的读端关闭、写端一直进行时 — 系统就会关闭管道(因为该管道无意义)发送13号信号SIGPIPE。也就在软件层面某些条件不满足而产生的信号!这里着重介绍一下alarm系统调用。
  5. 异常 :进程非法操作的时候,OS会发送信号!让进程崩溃(默认是终止进程,也可以进行捕捉异常信号。推荐终止进程!)

接下来我们来看看一下kill系统调用 、 alarm系统调用 、 异常。

kill系统调用

#include <signal.h>
#include <iostream>
#include <sys/types.h>
#include <unistd.h>
#include <string>

void handler(int sig)
{
    std::cout << "get sig:" << sig << std::endl;
}
// ./mykill 2 277015
int main(int argc , char* argv[])
{
    if(argc != 3) 
    {
        std::cerr << "Usage: " << argv[0] << " signum pid" << std::endl;
        return 1;
    }
    uint32_t sig = std::stoi(argv[1]);
    pid_t pid = std::stoi(argv[2]);
    //发送信号
    kill(pid , sig);

    return 0;
}

这样我们就被kill系统调用进行了一个封装!
我们使用一下来看:
在这里插入图片描述
成功向目标进行发送指定信号!

alarm系统调用

在这里插入图片描述
alarm系统调用会设置一个时间为seconds的“闹钟”,时间到了就会发送信号14) SIGALRM
我们设置一个闹钟看看

#include <signal.h>
#include <iostream>
#include <sys/types.h>
#include <unistd.h>
#include <string>
#include <cstdio>

int main()
{
    int cnt = 1;
    alarm(1); //设定1秒的闹钟
    while(true)
    {
        std::cout << cnt << std::endl;
        cnt++;
    }
}

运行看看:
在这里插入图片描述
1s后就停止了!因为14号信号对应的是term会进行终止!
在这里插入图片描述
我们也可以进行一个捕捉

#include <signal.h>
#include <iostream>
#include <sys/types.h>
#include <unistd.h>
#include <string>
#include <cstdio>

void handler(int sig)
{
    std::cout << "get sig:" << sig << std::endl;
    exit(1);
}

int main()
{
    int cnt = 1;
    signal(SIGALRM , handler);
    alarm(1); //设定1秒的闹钟
    while(true)
    {
        std::cout << cnt << std::endl;
        cnt++;
    }
}

在这里插入图片描述
成功捕捉到是14号信号!

如果我们不加上每次IO的输出,只在最后打印一次,我们会发现cnt会达到近10亿。
这就验证了IO是很慢的一个过程!因为IO的本质是向外设进行输出数据,外设的传输速度肯定比内存慢!!!

alarm的本质是通过时间戳来比对,在设置闹钟的那一刻,操作系统会获取当前时间戳,然后加上闹钟时间得到一个新的时间戳。在以后的运行中不断和系统时间戳进行比对,相等的时候是就是闹钟的结束时刻!!!

操作系统对闹钟的管理是也是通过内核数据结构struct alarm,并通过最小堆来进行。按照过期时间来排序,最上面的闹钟到时间了就进行pop,这样就可以进行一个管理!

闹钟的返回值是什么意义呢? 闹钟的返回值是上一个闹钟的剩余时间(alarm(0)表示取消闹钟)

注意闹钟默认只触发一次

void handler(int sig)
{
    alarm(1); //设定1秒的闹钟
    std::cout << "get sig:" << sig << std::endl;
}

int main()
{
    int cnt = 1;
    signal(SIGALRM , handler);
    alarm(1); //设定1秒的闹钟
    while(true)
    {
        std::cout << cnt << std::endl;
        cnt++;
    }
}

这样可以设置出一个一直在运行的闹钟:
在这里插入图片描述

异常

在学习C语言之初,一定接触过这样的代码:

int main()
{
    while(true)
    {
        int a = 100;
        a /= 0;

        // int *p = nullptr;
        // *p = 100;
    }
    return 0;
}

我们运行看看:
在这里插入图片描述
运行起来就崩溃了!
再来试试

int *p = nullptr;
*p = 100;

在这里插入图片描述
也崩溃了,那么为什么程序会崩溃呢???

因为程序非法访问,导致OS给进程发送信号,进程就崩溃了。来捕捉信号来证明一下:

void handler(int sig)
{
    std::cout << "get sig:" << sig << std::endl;
    exit(1);
}

int main()
{
    signal(SIGFPE , handler);
    while(true)
    {
        std::cout << "pid :" << getpid() << std::endl;
        int a = 100;
        a /= 0;
    }
}

运行看看:
在这里插入图片描述
这样就说明崩溃是因为收到了OS发送的信号!

那么OS是怎样知道进程进行非法操作的呢?以浮点数错误为例:

在CPU 运算中,数据是储存在内存中的。CPU可以进行算术运算和逻辑元素

CPU中有大量的寄存器,其中有一个eflag状态寄存器,在里面有一个溢出标记位,运算时出现溢出,溢出标记为为1,就证明出错了(10 / 0 首先会转换为加法运算,导致无限循环溢出)。这时候操作系统就要处理这种硬件问题!即向目标进程发送信号!

那为什么不退出就会一直发信号?

因为寄存器只有一套,但是寄存器里面的数据是属于每一个进程的 — 对应硬件上下文的保存与恢复。如果进程不退出,下一次调度的时候,对寄存器的数据进行恢复时,就会触发溢出标记位的错误,OS就会又一次发送信号!!!
这也就是为什么推荐终止进程 — 释放进程的上下文数据,包括溢出标记位数据和其他异常数据!

段错误也是硬件的问题,空指针无法通过页表(实际上是MMU内存管理模块进行操作)映射到物理地址,会发生错误!

CR2 - 控制寄存器2:
用于存储导致页错误的线性地址,当发生页错误异常时,CPU会自动将出错的线性地址加载到CR2中。
CR3 - 控制寄存器3:
包含页目录基址寄存器(PDBR),用于存储页目录表的物理地址,是分页机制的关键组成部分。
错误的地址会放入CR2中 , 触发故障。

core term

上面我们看到了许多信号的默认动作是core term 。他们有什么区别?
我们来看,进程退出时有一个status,其中的core dump字段我们来仔细看看。
在这里插入图片描述

  1. core : 异常终止,但是会形成一个debug文件(默认在云服务器是关闭的)
  2. term : 直接异常终止

debug文件是什么,我们一起看看:

在这里插入图片描述
首先默认是不能创建的,我们要进行一个修改:
在这里插入图片描述
接下来我们运行就会产生了一个core文件!!!,里面记录了错误信息!

Thanks♪(・ω・)ノ谢谢阅读!!!

下一篇文章见!!!

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

8086汇编 add指令学习

ADD&#xff0c;是Intel x86平台的汇编加法指令&#xff0c;MEM代指操作数为内存或寄存器&#xff0c;REG代指操作数为寄存器&#xff0c;IMM代指立即数&#xff0c;SEG代指操作数为段寄存器。 形式和示例如下&#xff1b; ADD MEM8,REG8 ADD DS:[BXSI],AL ADD MEM16,R…

20240615在WIN11下的串口调试助手的下载安装以及使用

20240615在WIN11下的串口调试助手的下载安装以及使用 2024/6/15 18:06 百度&#xff1a;串口调试助手 blob:https://apps.microsoft.com/df934d29-fd7a-4873-bb6b-a4ab5a7934c9 串口调试助手 Installer.exe 收发的LOG&#xff1a; rootok3588:/# ./uart_test /dev/ttyS0 11520…

MySQL数据操作与查询- 连接查询

一、引入 1、为什么需要使用连接查询&#xff1f; 查询信息的来源如果来自多张表&#xff0c;则必须对这些表进行连接查询。 2、连接查询的分类 内连接和外连接。 二、内连接 1、概述 将两张表的记录组合在一起&#xff0c;产生一个新的结果。 &#xff08;1&#xff09…

docker desktop for mac os如何使用本地代理

在macbook上弄了个代理&#xff0c;然后按照网上所说的去配代理 然后测试下 docker pull busybox 结果无反应&#xff0c;超时。我去&#xff01;&#xff01;&#xff01; 鼓捣了半天&#xff0c;看了docker官网&#xff0c;问了chatgpt &#xff0c;按照它们所说的试了下也没…

PCL 基于八叉树的去噪滤波

目录 一、算法原理1、原理概述2、参考文献二、代码实现三、结果展示1、滤波前2、滤波后本文由CSDN点云侠原创,原文链接。如果你不是在点云侠的博客中看到该文章,那么此处便是不要脸的爬虫与GPT。 一、算法原理 1、原理概述 使用八叉树对点云进行噪点删除的滤波方法实现。 …

C学习自学笔记-会陆续完善对应章节编程经典例子

C学习笔记 0>C语言概述 为什么学习C语言 1&#xff09;C的起源和发展------了解即可 B语言、C语言、C语言的产生地&#xff1a;都出自 美国贝尔实验室 2&#xff09;C的特点 优点&#xff1a;代码量小、速度快、功能强大 缺点&#xff1a;危险性高、开发周期长、可移植性…

UI学习--分栏控制器

UI学习 分栏控制器基础概念用法 分栏控制器高级高级属性 总结 分栏控制器基础 概念 分栏控制器可以理解为一个容器&#xff0c;可以容纳多个子视图控制器&#xff0c;并通过选项卡的方式进行切换。每个选项卡都与一个特定的视图控制器相关联&#xff0c;当用户点击不同的选项…

JUC并发编程第十三章——读写锁、邮戳锁

本章路线总纲 无锁——>独占锁——>读写锁——>邮戳锁 1 关于锁的面试题 你知道Java里面有那些锁你说说你用过的锁&#xff0c;锁饥饿问题是什么&#xff1f;有没有比读写锁更快的锁StampedLock知道吗&#xff1f;&#xff08;邮戳锁/票据锁&#xff09;ReentrantR…

「动态规划」如何求乘积最大子数组?

152. 乘积最大子数组https://leetcode.cn/problems/maximum-product-subarray/description/ 给你一个整数数组nums&#xff0c;请你找出数组中乘积最大的非空连续子数组&#xff08;该子数组中至少包含一个数字&#xff09;&#xff0c;并返回该子数组所对应的乘积。测试用例的…

WebGL学习【焕新计划】

WebGL基础 在正式进入webgl之前&#xff0c;我想有必要简单了解一下渲染管线&#xff0c;毕竟它贯穿webgl学习的整个过程。 渲染管线流程图&#xff1a; webgl着色器简单语法&#xff1a; 在GLSL&#xff08;OpenGL Shading Language&#xff09;中&#xff0c;常见的变量类…

企业化运维(4)_tomcat

###1.配置tomcat### 可以将tomcat部署在server2主机&#xff0c;与nginx主服务器分开&#xff0c;便于进行交互存储。 下载安装jdk与tomcat&#xff0c;并开启服务&#xff0c;便可以在浏览器进行访问。 [rootserver3 ~]# rpm -ivh jdk-8u121-linux-x64.rpm [rootserver3 ~]#…

Mybatis-Plus多种批量插入方案对比

背景 六月某日上线了一个日报表任务&#xff0c;因是第一次上线&#xff0c;故需要为历史所有日期都初始化一次报表数据 在执行过程中发现新增特别的慢&#xff1a;插入十万条左右的数据&#xff0c;SQL执行耗费高达三分多钟 因很早就听闻过mybatis-plus的[伪]批量新增的问题&…

万事开头难——Java实现俄罗斯小方块【第一步】

目录 技术实现&#xff1a; 1.初始化游戏窗口&#xff1b; 1.1 什么是窗口&#xff1a; 1.2 Swing 1.3 JFrame创建窗口&#xff1a; 1.3.1创建窗口的逻辑 1.3.2.设置简单的页面 1.3.3.优化 1.3.4.设置标题 1.4 创建游戏窗口 技术实现&#xff1a; 1.初始化游戏窗口&am…

【CT】LeetCode手撕—20. 有效的括号

题目 原题连接&#xff1a;20. 有效的括号 1- 思路 模式识别 模式1&#xff1a;括号左右匹配 ——> 借助栈来实现 ——> Deque<Character> deque new LinkedList<>()模式2&#xff1a;顺序匹配 ——> 用 if 判断 具体思路 1.遇到左括号 直接入栈相应…

ARM32开发--电源管理单元

知不足而奋进 望远山而前行 目录 文章目录 前言 学习目标 学习内容 PMU 电源域 VDD/VDDA域 备份域 1.2V域 省电模式 睡眠模式 深度睡眠模式 待机模式 几种模式总结 WFI和WFE指令 案例需求 模式初始化 源码 总结 前言 在嵌入式系统中&#xff0c;有效的电池管…

【AI基础】第六步:纯天然保姆喂饭级-安装并运行qwen2-7b

整体步骤类似于 【AI基础】第五步&#xff1a;纯天然保姆喂饭级-安装并运行chatglm3-6b-CSDN博客。 此系列文章列表&#xff1a; 【AI基础】概览 【AI基础】第一步&#xff1a;安装python开发环境-windows篇_下载安装ai环境python 【AI基础】第一步&#xff1a;安装python开发环…

面试题 17.05. 字母与数字

链接&#xff1a;. - 力扣&#xff08;LeetCode&#xff09; 题解&#xff1a;把字母看成1&#xff0c;把数字看为-1&#xff0c;将题目变为求的和为0的最长连续子数组 class Solution { public:vector<string> findLongestSubarray(vector<string>& array) …

MySQL查询练习题1.平均工资2.查询各部门的总薪水3.查询总薪水排名第二的部门4.查询姓名重复的员工信息5.查询各部门薪水大于900的男性员工的平均薪水

创建一个员工表emp&#xff0c;包含字段&#xff1a;姓名name&#xff0c;性别sex&#xff0c;部门depart&#xff0c;工资salary create table emp(name varchar(30) not null,sex varchar(30) not null,depart int not null,salary int not null); 插入数据打印为 mysql>…

Windows Server 2008 r2 IIS .NET

Windows Server 2008 r2 IIS .NET

2-2 基于matlab的变邻域

基于matlab的变邻域&#xff0c;含变惯性权重策略的自适应离散粒子群算法&#xff0c;适应函数是多式联运路径优化距离。有10城市、30城市、75城市三个案例。可直接运行。 2-2 路径规划 自适应离散粒子群算法 - 小红书 (xiaohongshu.com)