Linux - 信号概念 信号产生

news2024/9/20 15:05:39

Linux - 信号概念 & 信号产生

    • 信号概念
    • 信号产生
      • 软件信号
        • kill
        • raise
        • abort
        • alarm
      • 硬件信号
        • 键盘产生信号
        • 硬件中断


信号概念

信号是进程之间事件异步通知的一种方式

在Linux命令行中,我们可以通过ctrl + c来终止一个前台运行的进程,其实这就是一个发送信号的行为。我们按下ctrl + c是在shell进程中,而被终止的进程,是在前台运行的另外一个进程。因此信号是一种进程之间的通知方式。

可以通过指令kill -l来查询信号:

在这里插入图片描述

以上就是Linux中的全部信号,它们分为两个区间:[1, 31][34, 64],也就是说没有0,32,33这三个信号,虽然信号的最大编号为64,但实际上只有62个信号。

  • [1, 31]:这些信号称为非实时信号,当进程收到这些信号后,可以自己选择合适的时候处理
  • [34, 64]:这些信号称为实时信号,当进程收到这些信号后,必须立马处理

由于现在的操作系统基本都是分时操作系统,因此实时信号其实是不符合设计理念的,几乎用不到实时信号,本博客只讲解非实时信号

上图中,所有信号都是大写的单词,在C/C++中,一般来说就是大写的,其实信号名就是宏

那么进程收到信号后要怎么处理呢?

进程有三种处理信号的方式:

  1. 忽略此信号
  2. 执行信号的默认处理函数
  3. 执行信号的自定义处理函数,这种方式也成为信号捕捉

可以通过man 7 signal来查看信号的默认处理行为:

在开头,可以看到如下页面:

在这里插入图片描述

其中TermIgnCoreStopCont就是信号处理的默认行为,我简单翻译一下它们的作用:

  • Term:默认操作是终止进程
  • Ign:默认操作是忽略信号
  • Core:默认操作是终止进程并转储核心
  • Stop:默认操作是暂停进程
  • Cont:默认操作是,如果该进程当前已暂停,则继续该进程

以上五种

其中TermCore都是终止进程,不过Core会额外进行一个核心转储,这个会在博客后续讲解。

再往下翻阅,就可以看到每个信号的描述:

在这里插入图片描述

各列的含义如下:

  • Signal:信号的名称
  • Standard:该信号在哪一个标准中提出
  • Action:进程收到该信号后的默认处理行为
  • Comment:对信号的简单描述

对于刚才说的三种处理方式:

  1. 忽略此信号 -> 即 ActionIgn
  2. 执行信号的默认处理函数 -> 即Action 不为 Ign
  3. 执行信号的自定义处理函数,这种方式也成为`信号捕捉

那么我们又要如何自己定义信号处理函数呢?

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号信号,从而验证效果。

输出结果:

在这里插入图片描述

可以看到,我在hello world!期间按下了两次ctrl + C,本来ctrl + C发送2号信号时会直接终止进程,但是现在却是输出get sig 2了,因为我们将2号信号的行为改变了。


信号产生

简单讲解了一下信号的三种处理方式后,再来看看信号是如何产生的,在Linux中,信号主要有两种产生方式:软件条件硬件异常

软件信号

在 Linux 中,“软件条件” 发出的信号指的是由 进程自身或其他进程 产生的信号,而不是由硬件中断或其他外部事件触发的信号。

Linux中有多种系统调用可以发送信号,在此我讲解killraiseabortalarm四种接口,其中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指令发送信号,格式为:

kill -sig pid

其中sig为要发送的信号,pid为收到信号的进程pid

示例:

现在进程执行如下代码:

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

int main()
{
    signal(2, handler);

    while (true)
    {
        cout << "pid = " << getpid() << endl;
        sleep(1);
    }

    return 0;
}

进程先修改信号2的处理方式为handler,然后死循环输出自己的pid。随后我们在另外一个窗口通过kill -2 pid来发送(2) SIGINT信号。

输出结果:

在这里插入图片描述

左侧窗口中,进程输出自己的pid = 130988,在右侧窗口通过kill -2 130988向进程发送信号,输出一句get sig 2后退出。

其实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;
}

我们就在进程内部直接调用abort();,输出结果:

在这里插入图片描述

最后进程输出了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++

输出结果:

在这里插入图片描述

可以看到,计算到76451时,就收到了SIGALRM,终止进程了。


硬件信号

硬件信号指的是由 硬件事件 触发的信号,而不是由软件代码逻辑控制的。

具体来说,以下情况属于硬件条件发出的信号:

  • 中断: 硬件设备,例如键盘、鼠标、网络接口等,在发生事件时会向 CPU 发送中断信号,例如键盘按键按下、网络数据包到达等。
  • 异常: CPU 在执行指令过程中,如果遇到错误情况,例如除以零、内存访问错误等,会产生异常信号。
  • 时钟中断: 系统定时器会定期向 CPU 发送时钟中断信号,用于调度进程和执行定时任务。
键盘产生信号

通过键盘发送信号是最简单的信号发送方式,最常用的有ctrl + Cctrl + \

  • ctrl + C:向前台进程发送(2) SIGINT信号,效果为直接终止进程
  • ctrl + \:向前台进程发送(3) SIGQUIT信号,效果为直接终止进程

示例:

在这里插入图片描述

该进程每隔一秒输出Hello world,第一次执行进程,我按下ctrl + C后直接终止了进程,第二次按下ctrl + \输出了一个Quit后终止进程。


硬件中断

硬件给进程发送信号的本质,其实是通过硬件中断

硬件中断是指硬件设备向 CPU 发送的信号,通知 CPU 有事件发生需要处理。

想象一下,你的电脑就像一个忙碌的办公室,CPU 就像办公室的经理,负责处理各种任务。而键盘、鼠标、网卡等硬件设备就像办公室的员工,需要向经理汇报工作进度或请求帮助。

当一个员工需要向经理汇报工作进度时,他会敲响经理办公室的门,发出一个“中断”信号。经理收到这个信号后,会暂停当前的工作,转而去处理员工的请求。

硬件中断也是类似的机制。当一个硬件设备需要向 CPU 汇报事件发生时,它会向 CPU 发送一个中断信号。CPU 接收这个信号后,会暂停当前执行的任务,转而去处理硬件设备的请求

看到以下代码:

int main()
{
    int a = 5 / 0;

    return 0;
}

这是一个很经典的除零错误,如果我们强行运行这个进程,会报出如下错误:

在这里插入图片描述

发送了错误Floating point exception,本质上是收到了信号(8) SIGFPE,那么为什么除零会发送这个信号呢?其实本质上是发生了硬件中断。

如下图:

在这里插入图片描述

在以上过程中,各个部分发挥作用如下:

  • CPUCPU是一个硬件,其要处理计算,而5/0这个计算过程就是在CPU中完成的,当CPU检测到这个计算中0做了除数,于是对自己发起硬件中断
  • 操作系统:操作系统检测到硬件中断后,跳转到中断处理程序中断处理程序Linux内核的一部分),然后检测该硬件中断是什么原因,发现是因为除零错误,于是给进程发送(8) SIGPFE信号
  • 进程:进程收到操作系统发来的(8) SIGPFE信号后,执行相应的处理措施

这就是硬件中断发送信号的过程,其实发送信号本质还是操作系统来发送,而硬件发现异常后,只是通知操作系统去发送信号。

再比如说我们之前的ctrl + C按键发送(2) SIGINT信号,本质也是硬件中断,当我们从键盘输入了数据后,键盘向CPU发出硬件中断,随后CPU去执行操作系统中的硬件中断程序,发现是用户按下了ctrl + C,于是操作系统向进程发送(2) SIGINT信号。

再看到这样的代码:

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

int main()
{
    signal(8, handler);

    int a = 5 / 0;

    return 0;
}

我们已经知道,除0错误是(8) SIGFPE信号,现在我们把其对应的处理方式改为输出一条语句,我们看看会发生什么:

在这里插入图片描述

可以看到,进程现在在不断的输出get sig: 8,可是我们只发生了一次int a = 5 / 0,为什么会产生这么多信号?

现在先想一想,为什么我们进程会收到(8) SIGFPE信号,这是因为CPU检测到5 / 0的错误,发生了硬件中断。请问我们调用了handler函数之后,CPU内部的5 / 0被处理了吗?答案是没有的!

也就是说,CPU会被一直卡在5 / 0CPU不知道怎么计算这个表达式,于是一直硬件中断,操作系统就一直发送(8) SIGFPE信号,所以进程就一直执行handler函数了。

与软件条件信号相比,硬件条件信号具有以下特点:

  • 由硬件触发: 硬件条件信号的产生是由硬件事件触发的,而不是由代码逻辑决定的
  • 不可控性: 硬件条件信号的产生通常无法被进程控制,例如硬件设备的故障、网络连接中断等。
  • 不可预测性: 硬件条件信号的产生通常是不可预测的,因为它们是由硬件事件触发的。

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

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

相关文章

AI全栈工程师的新舞台:Coze(扣子)

前言 在当前科技飞速发展的背景下&#xff0c;Coze作为一款引领潮流的AI应用平台&#xff0c;正以破竹之势重塑着我们对于智能应用的认知。Coze不仅仅是一个工具&#xff0c;它是一个集合了前沿AI技术、高效开发环境与创意无限的应用生态于一体的创新平台&#xff0c;旨在让每…

ctfshow-web入门-命令执行(web53-web55)

目录 1、web53 2、web54 3、web55 1、web53 这里的代码有点不一样&#xff0c;说一下这两种的区别&#xff1a; &#xff08;1&#xff09;直接执行 system($c); system($c);这种方式会直接执行命令 $c 并将命令的输出直接发送到标准输出&#xff08;通常是浏览器&#xff…

如何理解external

external 函数应该只被外部函数调用但也可以被内部调用&#xff0c;但是这种内部调用也是有外部调用机制&#xff0c;即新产生message! 例子1 // SPDX-License-Identifier: GPL-3.0pragma solidity >0.8.2 <0.9.0;contract ExternalDemo{address public caller;functi…

【Activiti7系列】基于Spring Security的Activiti7工作流管理系统简介及实现(附源码)(下篇)

作者&#xff1a;后端小肥肠 上篇&#xff1a;【Activiti7系列】基于Spring Security的Activiti7工作流管理系统简介及实现&#xff08;上篇&#xff09;_spring security activiti7-CSDN博客 目录 1.前言 2. 核心代码 2.1. 流程定义模型管理 2.1.1. 新增流程定义模型数据 …

【qsort函数】

前言 我们要学习qsort函数并利用冒泡函数仿照qsort函数 首先我们要了解一下qsort&#xff08;快速排序&#xff09; 这是函数的的基本参数 void qsort (void* base, size_t num, size_t size,int (*compar)(const void*,const void*)); 简单解释一下 base&#xff1a;指向…

自动化搭建专属 AI 绘图服务

通义万相AIGC技术已经比较成熟&#xff0c;结合阿里云的计算和存储产品可以方便的搭建自己专属的 AI 绘图服务。例如《创意加速器&#xff1a;AI 绘画创作》这个解决方案&#xff0c;利用阿里自研的通义万相AIGC技术在 Web 服务中实现先进的图像生成。 AI 绘画服务搭建步骤 从…

【文档智能 RAG】RAG增强之路:增强PDF解析并结构化技术路线方案及思路

前言 现阶段&#xff0c;尽管大模型在生成式问答上取得了很大的成功&#xff0c;但由于大部分的数据都是私有数据&#xff0c;大模型的训练及微调成本非常高&#xff0c;RAG的方式逐渐成为落地应用的一种重要的选择方式。然而&#xff0c;如何准确的对文档进行划分chunks&…

Golang的协程调度器GMP

目录 GMP 含义 设计策略 全局队列 P的本地队列 GMP模型以及场景过程 场景一 场景2 场景三 场景四 场景五 场景六 GMP 含义 协程调度器&#xff0c;它包含了运行协程的资源&#xff0c;如果线程想运行协程&#xff0c;必须先获取P&#xff0c;P中还包含了可运行的G…

时序数据库是Niche Market吗?

引言 DB-Engines的流行程度排行从其评估标准[4]可以看出完全不能够做为市场规模的评估标准。甚至于在知道市场规模后可以用这个排行作为一个避雷手册。毕竟现存市场小&#xff0c;可预见增长规模小&#xff0c;竞争大&#xff0c;创新不足&#xff0c;那只能卷价格&#xff0c…

01、Linux网络设置

目录 1.1 查看及测试网络 1.1.1 查看网络配置 1、查看网络接口地址 2、查看主机状态 3、查看路由表条目 4、查看网络连接qing 1.1.2 测试网络连接 1.测试网络连接 2.跟踪数据包的路由路径 3.测试DNS域名解析 1.2 设置网络地址参数 1.2.1 使用网络配置命令 1.修改网卡…

C# MES通信从入门到精通(11)——C#如何使用Json字符串

前言 我们在开发上位机软件的过程中&#xff0c;经常需要和Mes系统进行数据交互&#xff0c;并且最常用的数据格式是Json&#xff0c;本文就是详细介绍Json格式的类型&#xff0c;以及我们在与mes系统进行交互时如何组织Json数据。 1、在C#中如何调用Json 在C#中调用Json相关…

【题解】—— LeetCode一周小结23

&#x1f31f;欢迎来到 我的博客 —— 探索技术的无限可能&#xff01; &#x1f31f;博客的简介&#xff08;文章目录&#xff09; 【题解】—— 每日一道题目栏 上接&#xff1a;【题解】—— LeetCode一周小结22 3.分糖果 II 题目链接&#xff1a;1103. 分糖果 II 排排坐…

【漏洞复现】用友NC pagesServlet SQL注入漏洞(XVE-2024-13067)

0x01 产品简介 用友NC是由用友公司开发的一套面向大型企业和集团型企业的管理软件产品系列。这一系列产品基于全球最新的互联网技术、云计算技术和移动应用技术&#xff0c;旨在帮助企业创新管理模式、引领商业变革。 0x02 漏洞概述 用友NC /portal/pt/servlet/pagesServlet…

Springboot校园美食推荐系统的开发-计算机毕业设计源码44555

摘要 随着人们生活水平的提高&#xff0c;人们对美食的要求也越来越高&#xff0c;对各类美食信息需求越来越大。因此&#xff0c;结合计算机快速发展、普及&#xff0c;在此基础上制作一个页面简单、美观,功能实用的校园美食推荐系统势在必行&#xff0c;满足用户分享美食的需…

Spring-Security(二)OAuth2认证详解(持续更新)

Spring Security & Oauth2系列&#xff1a; Spring Security&#xff08;一&#xff09; 源码分析及认证流程 Spring Security&#xff08;二&#xff09;OAuth2认证详解及自定义异常处理 文章目录 1、OAuth2.0 简介1.1 OAuth2.0 相关名词解释1.2 四种授权模式 1.3 、OAu…

QT 信号和槽 信号关联到信号示例 信号除了可以绑定槽以外,信号还可以绑定信号

信号除了可以关联到槽函数&#xff0c;还可以关联到类型匹配的信号&#xff0c;实现信号的接力触发。上个示例中因为 clicked 信号没有参数&#xff0c;而 SendMsg 信号有参数&#xff0c;所以不方便直接关联。本小节示范一个信号到信号的关联&#xff0c;将按钮的 clicked 信号…

Python 深度探讨 *args

点击下方卡片&#xff0c;关注“小白玩转Python”公众号 作为Python中最独特的语法之一&#xff0c;*args 在编程过程中给我们带来了很多灵活性和便利性。我认为它们反映了“Pythonic”和“Python之禅”。然而&#xff0c;我发现它们对于学习者&#xff08;尤其是初学者&#x…

DeepSpeed Learning Rate Scheduler

Learning Rate Range Test (LRRT) 训练试跑&#xff0c;该lr scheduler从小到大增长lr&#xff0c;同时记录下validatin loss&#xff1b;人来观察在训练多少step之后&#xff0c;loss崩掉&#xff08;diverge)了&#xff0c;进而为真正跑训练&#xff0c;挑选合适的lr区间&…

一、Electron 环境初步搭建

新建一个文件夹&#xff0c;然后进行 npm init -y 进行初始化&#xff0c;然后我们在进行 npm i electron --save-dev , 此时我们按照官网的教程进行一个初步的搭建&#xff0c; 1.在 package.json 文件进行修改 {"name": "electron-ui","version…

嵌入式应用之FIFO模块原理与实现

FIFO介绍与原理 FIFO是First-In First-Out的缩写&#xff0c;它是一个具有先入先出特点的缓冲区。FIFO在嵌入式应用的非常广泛&#xff0c;可以说有数据收发的地方&#xff0c;基本就有FIFO的存在。或者为了降低CPU负担&#xff0c;提高数据处理效率&#xff0c;可以在积累到一…