Linux信号(产生)

news2024/10/6 14:34:48

个人主页:Lei宝啊 

愿所有美好如期而遇


目录

信号是什么? 

为什么要有信号?

信号是如何产生的?

kill命令

键盘产生信号

系统调用

kill系统调用

raise函数

abort函数

自制kill命令

​编辑

软件条件

举例一:

举例二:

举例三:

异常

除0操作被发送信号原理

野指针问题使程序崩溃原理

异常信号的自定义

信号产生的几个问题

问题一:操作系统怎么知道键盘按键被按下,然后去读取字符或者控制命令(ctrl+c等)?

问题二:信号产生后写入task_struct中,但是task_struct是内核数据结构,用户可以写吗? 


信号是什么? 

信号是Linux提供的使用户(进程)给其他进程发送异步信息的一种方式。

我们可以先给出这样几个结论:

  1. 在信号还没有发送给当前进程时,当前进程已经知道当一个信号发送过来时,该如何处理这个信号
  2. 信号应该被提前设置好,而进程中有识别信号的方式,直到如何进行处理。
  3. 信号在发送过来时,如果进程在执行重要的代码时,信号应当进行临时保存。
  4. 信号到来时,不一定要立即处理。
  5. 信号的产生是随机的,我们无法准确预测信号何时发送过来,所以信号是异步发送的。
  • 同步:在同步操作中,一个任务需要等待前一个任务完成后才能开始,也就是说,任务必须按照某种顺序一步一步来执行,后一个任务依赖前一个任务的结果
  • 异步:在异步操作中,任务可以不必等待前一个任务完成就立即开始,异步操作不依赖于之前任务是否完成,只是简单地启动任务,然后让操作系统在后台处理任务,它不会阻塞程序的后续执行

因为我们不知道信号是何时发送的,并且信号的发送并不依赖于要给到信号的进程,所以信号的发送不必等待进程执行完一个任务,而是在进程执行过程中,进程执行他的,信号发送他的,各自做各自的事情,两者并发执行。

我们可以来看看常见的信号

后面的信号这里剪掉了,他们是实时信号,我们要介绍的是分时操作系统,所以不对实时信号做解释。

信号的名字和编号都可以标识信号,他的名字就是宏,没有0号信号,也没有32,33号信号,对信号的学习涉及三个阶段:产生,保存,处理,本节我们着重介绍信号的产生。

简单介绍信号的处理方式

  • 默认动作,即默认设定的动作。
  • 自定义动作,用户设定的动作。
  • 忽略。

自定义动作通过signal系统调用实现:

用法我们在后面会谈到。 

为什么要有信号?

因为系统要求进程能够有随时响应外部信号的能力,能够做出反应,停止或者终止等等。

信号是如何产生的?

kill命令

kill -9 pid  杀掉一个进程        (也可以kill -SIGKILL pid)

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

using namespace std;

int main()
{

    while(true)
    {
        cout << "进程pid: " << getpid() << endl;
        sleep(1);
    }
    
    return 0;
}

kill -2 pid  终止一个进程 

键盘产生信号

比如ctrl + c, ctrl + \,都可以使一个进程退出,这里我们不卖关子,直接挑明,其实这两个控制命令会被解释成为信号,ctrl + c会被解释为2号信号,也就是终止一个进程,ctrl + \会被解释为3号信号,也是终止进程。

我们这里总是谈到几号信号可以终止进程,我们是怎么知道的?手册。

man 7 signal  一直往下翻就可以看到

我们还可以看到的同样是终止进程,但是他们的动作有Term和Core,其他信号的动作也有Stop,Ign(忽略),我们后面会着重谈到Term和Core的区别。

系统调用

kill系统调用

int kill(pid_t pid, int sig);     sig可以填编号,亦可以写宏名

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

using namespace std;

int main()
{
    int cnt = 0;

    while(true)
    {
        if(cnt++ == 3) kill(getpid(),9);
        
        cout << "进程pid: " << getpid() << endl;
        sleep(1);
    }
    
    return 0;
}

raise函数

int raise(int sig);    给当前进程进行使用

kill系统调用可以使一个进程给其他进程发送信号,而raise只能用于当前进程,给自己发送信号。

int main()
{
    int cnt = 0;

    while(true)
    {
        if(cnt++ == 3) raise(9);

        cout << "进程pid: " << getpid() << endl;
        sleep(1);
    }
    
    return 0;
}

abort函数

这个也是进程终止,但是主要是用于出现异常时使用。

我们不难猜到,后面两个函数底层封装了kill系统调用。

自制kill命令
#include <iostream>
#include <signal.h>
#include <unistd.h>
#include <sys/types.h>
#include <cerrno>
#include <cstring>

using namespace std;

int main(int argc, char* argv[])
{

    if(argc != 3)
    {
        cout << "Usage: " << argv[0] << "-信号 pid" << endl; 
    }

    int sig = atoi(argv[1] + 1);
    int pid = atoi(argv[2]);

    int val = kill(pid,sig);
    if(val < 0)
    {
        cout << "errno: " << errno << strerror(errno) << endl;
    }

    return 0;
}

软件条件

管道中有这样一种情况:读端关闭后,写端进程会被终止,被发送信号为:13 SIGPIPE。什么叫做软件条件呢?就是说,首先他是个软件,什么是软件?就是我们编写的代码,管道是不是编写的代码? 是,所以他是软件,那么他被发送信号的条件是什么?读端关闭,所以叫做软件条件。

这里我们再介绍一种软件条件:闹钟

调用alarm函数可以设定一个闹钟,也就是告诉内核在seconds秒之后给当前进程发SIGALRM信号, 该信号的默认处理动作是终止当前进程 。

这个函数的返回值是0或者是以前设定的闹钟时间还余下的秒数,我们举例来理解。

举例一:
#include <iostream>
#include <signal.h>
#include <unistd.h>
#include <sys/types.h>

using namespace std;

int main()
{
    alarm(1);

    int cnt = 0;
    while(true)
    {
        cout << "cnt: " << cnt++ << endl; 
    }

    return 0;
}

我们发现cnt仅仅加到6万多,是不是太少了?我们现在来介绍自定义处理信号:

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

using namespace std;

//这里不写exit进程不退的原因是while循环,
//进程只收到了一次信号(不同于异常)。
int cnt = 0;
void handler(int signum)
{
    cout << "cnt: " << cnt << endl;
    exit(0);
}

int main()
{
    signal(SIGALRM,handler);
    alarm(1);
  
    while(true)
    {
        cnt++;
    }

    return 0;
}

 signa仅仅是设置,只有在进程收到信号时,才会去执行这个自定义处理方法。

从这里也可以看出,IO的消耗还是很大的。

举例二:

我们能不能让闹钟一直响,而不是响一次就不响了?

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

using namespace std;

void handler(int signum)
{
    cout << "闹钟响了,梦醒了" << endl;
    alarm(2);
}

int main()
{
    signal(SIGALRM,handler);
    alarm(2);
  
    while(true)
    {
        cout << "恋爱循环" << endl;
        sleep(1);
    }

    return 0;
}

逻辑是这样的,首先我们signal自定义设置信号处理方式,然后调用alarm,2秒后调用自定义方法,而自定义方法中又设了闹钟,这样的方式,我们可以让闹钟响我们想要的时长。 

举例三:

这里我们来解释返回值问题。

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

using namespace std;

void handler(int signum)
{
    cout << "闹钟响了,梦醒了" << endl;
    int handler_val = alarm(2);
    cout << "handler_val: " << handler_val << endl;
}

int main()
{
    signal(SIGALRM,handler);
    alarm(50);
  
    while(true)
    {
        cout << "恋爱循环" << getpid() << endl;
        sleep(1);
    }

    return 0;
}

我们设置了50秒后响闹钟,现在我们提前使闹钟响,那么他会返回距旧闹钟响的剩余时间,并按照新的设置去响闹钟。 如果正常响闹钟,剩余时间就是0。

 那么怎么理解闹钟也是个软件条件呢?

alarm是个系统调用,闹钟的设定是在操作系统内部的,并且操作系统内部可能不止一个闹钟,那么操作系统该如何管理这些闹钟呢?先描述,再组织,所以在操作系统内部,应该是这样一个结构:

并且以按照过期时间以最小堆的方式进行组织,过期时间是按照当前时间的时间戳+设置的时间,当过期时间到了的时候,操作系统会找到对应的进程的进程控制块发送SIGALRM信号。

所以我们说,alarm是软件条件,条件就是过期时间。 

异常

我们的除0操作,野指针问题,都是异常,除0操作会发送8号信号SIGFPE,野指针操作会发送11号信号SIGSEGV,也就是段错误。

除0操作被发送信号原理

我们需要介绍到,CPU中有一套寄存器,其中有一个寄存器叫做标志寄存器,里面有一个溢出标志位,当计算的值过大溢出,这个标志位就会被置1,如果计算结果可信,那么就是0。

当这个标志位被标记为1时,会给到操作系统,操作系统就会向这个进程发送SIGFPE信号。

野指针问题使程序崩溃原理

CPU中存在一套寄存器,其中CR2记录页故障地址,CR3保存页表地址,MMU是内存管理单元,他和操作系统一起将虚拟地址转换为物理地址。

我们上面想要对虚拟地址为0的空间写入6这个值,我们需要主要到的是,虚拟地址空间中,低地址的位置通常都只有读权限,不允许修改,甚至虚拟地址空间的范围不包括0地址,那么我们想用0号虚拟地址去转换一个物理地址,这是不合法的,于是转换失败,失败信息填入CR2,交由操作系统,操作系统给当前进程发送SIGSEGV信号。

异常信号的自定义

异常信号的自定义,如果不写退出函数,那么可能会发生这样的事情:

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

using namespace std;

void handler(int signum)
{
    cout << "handler" << endl;
    sleep(1);
}

int main()
{

    signal(SIGFPE,handler);

    int a = 10;
    a /= 0;

    return 0;
}

也就是说,程序不会退出,会一直执行handler,这是因为,CPU在执行进程代码时,寄存器里的那些数据是属于进程的,而CPU执行出了异常,对异常进行了捕捉,而我们自定义的异常处理方法只是打印一句话,异常信息没有被处理,那么当前进程就不退出了,当前进程的上下文就要进行保留,操作系统还要正常跑,还要进行正常的进程调度,那么就要进行对进程上下文的保护和恢复,当下一次对该进程恢复数据时,标志位寄存器上的异常仍然在,那么继续捕捉,打印一句话,异常未被处理,进程仍然不退,所以就这样,一直进行打印。

而怎么处理异常信息呢?我们直接exit使进程退出即可,那么寄存器里关于这个进程的数据和异常信息也就废弃了,操作系统也就不会维护了。

这样的道理在野指针上也是同样的。

信号产生的几个问题

问题一:操作系统怎么知道键盘按键被按下,然后去读取字符或者控制命令(ctrl+c等)?

首先,键盘是硬件,这些硬件在主板上是和CPU连接的,即使是外置,也提供了USB口。

再一个,CPU上有一些针脚,每个针脚有一个编号,我们叫做中断号,键盘对应的中断号是2,那么,在我们按下键盘上的按键时,会向CPU上的针脚发送高电频,于是CPU就将2号编号写到reo寄存器中,操作系统在识别到后,就知道键盘被按下了。

操作系统在开机时,会维护一个中断向量表,其实就是一个函数指针数组,里面也就是各种方法,在操作系统识别到键盘被按下后,通过中断号,在中断向量表中找到读取键盘数据的方法,将数据读上来后进行判定,如果是普通字符,那么就写到键盘struct file的内核缓冲区,再由用户通过进程的方法读上去,如果是控制命令,就由操作系统解释成信号发送给进程。

什么叫做解释成信号?我们一开始说过常见的31个信号,我们通过task_struct中的一个位图来进行维护,解释成信号也就是说写进这个位图中进行保存,本章节我们简单说一说信号保存和处理,后面我们会做详细介绍。

问题二:信号产生后写入task_struct中,但是task_struct是内核数据结构,用户可以写吗? 

用户当然不可以写,这种事情只能由操作系统来完成,但是用户也想写怎么办?那么就由操作系统提供系统调用。

现在,我们来看看产生信号的五种方式:

kill命令,我们也实现过,也是调用的kill系统调用;软件条件,不管是alarm还是管道,调用的也都是系统调用,异常也是如此。也就是说,信号的产生,本质上就是一种:系统调用。


Core和Term的区别下节在信号的保存会解释。 

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

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

相关文章

C++ :设计模式实现

文章目录 原则单一职责原则开闭原则依赖倒置原则接口隔离原则里氏替换原则 设计模式单例模式观察者模式策略模式代理模式 原则 单一职责原则 定义&#xff1a; 即一个类只负责一项职责 问题&#xff1a; 类 T 负责两个不同的职责&#xff1a;职责 P1&#xff0c;职责 P2。当…

大数据第六天

这里写目录标题 问题解决问题查询插入(时间慢)练习sql数据清理 问题 FAILED: ParseException line 1:16 mismatched input ‘input’ expecting INPATH near ‘local’ in load statement MismatchedTokenException(24!155) 加载数据的时候出现了这个错误&#xff0c;我们解释…

【六十】【算法分析与设计】用一道题目解决dfs深度优先遍历,dfs中节点信息,dfs递归函数模板进入前维护出去前回溯,唯一解的剪枝飞升返回值true

路径之谜 题目描述 小明冒充X星球的骑士,进入了一个奇怪的城堡。 城堡里边什么都没有,只有方形石头铺成的地面。 假设城堡地面是nn个方格。如下图所示。 按习俗,骑士要从西北角走到东南角。可以横向或纵向移动,但不能斜着音走,也不能跳跃。每走到一个新方格,就要向正北 方和正西…

短信视频提取批量工具,免COOKIE,博主视频下载抓取,爬虫

痛点&#xff1a;关于看了好多市面的软件&#xff0c;必须要先登录自己的Dy号才能 然后找到自己的COOKIE 放入软件才可以继续搜索&#xff0c;并且无法避免长时间使用 会导致无法正常显示页面的问题。 有没有一种方法 直接可以使用软件&#xff0c;不用设置的COOKIE的方法呢 …

对于地理空间数据,PostGIS扩展如何在PostgreSQL中存储和查询地理信息?

文章目录 一、PostGIS扩展简介二、PostGIS存储地理空间数据1. 创建空间数据表2. 插入空间数据 三、PostGIS查询地理空间数据1. 查询指定范围内的地理空间数据2. 计算地理空间数据之间的距离3. 对地理空间数据进行缓冲区分析 四、总结 地理空间数据是指描述地球表面物体位置、形…

开源社区与开发者的故事

开源社区与开发者的故事 什么是开源社区你参加开源社区的主要目的你是否在开源社区中贡献&#xff0c;或者开源自己的项目&#xff1f;你认为个人开发者是否应该从开源中获利&#xff1f;如果是&#xff0c;该如何获利&#xff1f; 今天要谈及的主题是开源社区&#xff0c;那么…

2024年新算法-牛顿-拉夫逊优化算法(NRBO)优化BP神经网络回归预测

亮点&#xff1a; 输出多个评价指标&#xff1a;R2&#xff0c;RMSE&#xff0c;MSE&#xff0c;MAPE和MAE 满足需求&#xff0c;分开运行和对比的都有对应的主函数&#xff1a;main_BP, main_NRBO, main_BPvsBP_NRBO&#xff0c;并且详细中文注释 方便快捷&#xff1a;替换…

打破企业差旅管理困局,让金融CEO眼前一亮的出行方案

在国内券商投行部工作是怎样一种体验&#xff1f; “长期出差&#xff0c;而且出长差&#xff0c;时常让人有漂泊的孤独感。”这是某问答平台上的高赞回答的第一条。 对金融人来说&#xff0c;说走就走的旅行可能根本没有什么吸引力&#xff0c;时刻准备着说走就走的出差才是生…

MVCC的执行原理

MVCC的执行原理 MVCC简介事务的隔离级别MVCC作用当前读和快照读MVCC实现原理Undo LogUndo Log 版本链Read View判断方法判断规则 小结 MVCC简介 MVCC&#xff08;Multi-Version Concurrency Control&#xff09;是一种并发控制机制&#xff0c;用于解决数据库并发访问中&#…

pyqt 动态更换表头和数据

目录 pyqt 动态更换表头和数据代码 效果图&#xff1a; pyqt 动态更换表头和数据代码 from PyQt5.QtGui import QColor, QBrush from PyQt5.QtWidgets import QApplication, QTableWidget, QVBoxLayout, QWidget, QPushButton, QTableWidgetItemclass Example(QWidget):def _…

如何诊断并解决PostgreSQL中的磁盘空间不足问题?

文章目录 诊断磁盘空间不足问题1. 检查服务器磁盘空间2. 检查PostgreSQL数据目录大小3. 检查PostgreSQL中的大表和大对象 解决磁盘空间不足问题1. 清理不必要的文件和日志2. 清理或压缩大表和大对象3. 扩展磁盘容量4. 优化数据库配置和查询 在使用PostgreSQL数据库时&#xff0…

华为云实验 -- 对云硬盘数据盘进行备份

文章目录 备份Linux系统备份1.购买Linux操作系统的ESC(云服务器)2.挂载数据盘--初始化--分区--格式化2.1.点击"远程登录"a.查看/dev/vdb数据盘b.新建主分区/dev/vdb1 2.2.查看新建分区大小,分区格式信息a.确定之前的分区操作是否正确b.确认完成后&#xff0c;将分区结…

【MATLAB源码-第32期】基于matlab的通信及雷达中常用伪随机码m序列的仿真。

操作环境&#xff1a; MATLAB 2022a 1、算法描述 M序列&#xff0c;也称为最大长度序列或者伪随机序列&#xff0c;是一种特殊的二进制序列。它的特点是在有限的长度内&#xff0c;尽管它是伪随机的&#xff0c;但它会在特定的周期内不重复地循环。 在数学上&#xff0c;M序…

利用fft算法重写公式并理解频率和像素变化率的关系(完美解决问题)

算法我就不贴了。算法就是算法导论的内容。 我直接写推导过程。 假设变化率为f(n1)-f(n) 首先计算二进制数&#xff0c;这里我假设为3位二进制。 例如:f(5)-f(4)&#xff0c; 5和4的二进制为101,100。所以逆序数为101&#xff0c;001 101对应的频率为5, 001对应的频率为1…

力扣HOT100 - 236. 二叉树的最近公共祖先

解题思路&#xff1a; dfs 节点p,q异侧时&#xff0c;节点root为它们的公共祖先。 class Solution {public TreeNode lowestCommonAncestor(TreeNode root, TreeNode p, TreeNode q) {if (root null || p root || q root) return root;TreeNode left lowestCommonAncest…

max各种相机导出到ue4匹配镜头的工具集

总览 rollout export_UE4Cam_v2 "导出UE4Cam_v2:半自动" width:200 height:120(HyperLink explain "在打开的max文件中使用" pos:[25,12] width:200 height:15 color:(color 255 155 0) GroupBox grp1 "要导出的相机名" pos:[5,28] width:179 …

NeRF in the Wild: Neural Radiance Fields for Unconstrained Photo Collections

NeRF in the Wild: Neural Radiance Fields for Unconstrained Photo Collections(野外的 NERF: 用于无约束照片采集的神经辐射场&#xff09; Abstract 我们提出了一种基于学习的方法来合成新的视图的复杂场景使用只有非结构化的收集野生照片。我们建立在神经辐射场(neRF)的…

深度学习算法简介(一)

目录 ⛳️推荐 前言 1、深度神经网络&#xff08;DNN&#xff09; 2、卷积神经网络&#xff08;CNN&#xff09; 3、残差网络&#xff08;ResNet&#xff09; 4、LSTM&#xff08;长短时记忆网络&#xff09; 5、Word2Vec 6、Transformer 7、生成对抗网络&#xff08;…

MySQL常见的约束

什么是约束&#xff1f; 限制&#xff0c;限制我们表中的数据&#xff0c;保证添加到数据表中的数据准确和可靠性&#xff01;凡是不符合约束的数据&#xff0c;插入时就会失败&#xff0c;插入不进去的&#xff01; 比如&#xff1a;学生信息表中&#xff0c;学号就会约束不…

【IC设计】奇数分频与偶数分频 电路设计(含讲解、RTL代码、Testbench代码)

文章目录 原理分析实现和仿真偶数分频的电路RTL代码偶数分频的电路Testbench代码偶数分频的电路仿真波形占空比为50%的三分频电路RTL代码占空比为50%的三分频电路Testbench代码占空比为50%的三分频电路仿真波形 参考资料 原理分析 分频电路是将给定clk时钟信号频率降低为div_c…