Linux: 进程信号初识

news2025/4/8 17:50:50

目录

一 前言

二 信号的感性认识

三 信号处理常见方式

四 系统信号列表

 五 信号的保存

六 信号的产生

1. 通过终端按键产生信号 

2. 通过系统调用向进程发送信号

 3. 硬件异常产生信号

4. 软件条件产生信号 


一 前言

  在Linux操作系统中,进程信号是一个非常重要的概念。我们在前面的文章中已经见识过了进程信号了,比如我们在进程的一章中,尝试过向进程发送9号信号来终止进程(kill -9 pid)。所以对于用户来说,我们可以通过向进程发送特定的信号使得进程完成某些指定的动作。


二 信号的感性认识

我们在看到红灯的时候就知道需要停下,看到绿灯的时候就知道可以通过,那为什么我们在接收到这种信号之后,就知道做出相应的动作呢?这是因为我们能识别出不同的灯对应的含义,所以我们一旦接收到信号,就知道该怎么做了。

如同生活中的信号,Linux中的信号也是这样的,我们在给进程发送信号的时候,这个信号一定是进程能识别的,且具有一定含义,这样进程才能做出相应的行为。所以

  • 在进程收到信号前,它就有着处理信号的能力
  • 无论有没有接收到信号,进程都需要有识别信号的能力
  • 无论有没有接收到信号,进程都需要有着处理信号的能力,即需要该信号表示的含义或者功能。

三 信号处理常见方式

生活中,我们是如何处理信号的?举个快递的例子,快递员给你打电话说快递到了让你去取,此时一般会有以下几种处理方式:

  1. 立即去取
  2. 由于手头有事,让快递员等一下再去取
  3. 暂时走不开,让快递员将快递放在固定点自己找时间去取

即在快递到的时候,我们大概率是在忙着的,也就是说快递到了这件事和我们当前的状态是异步发生的。

事实上,进程信号也是一样的,我们在给进程发送信号的时候,是异步通知的。进程在收到信号的时候大概率是在忙的,所以进程此时就可以选择对信号处理方式

  1. 执行该信号的默认处理动作。
  2. 忽略此信号。
  3. 接收到信号,自定义处理,即用户接收到信号之后,自定义信号的处理动作

四 系统信号列表

用   kill -l   命令可以察看系统定义的信号列表

 可以看到系统一共存在着64种信号,其中1~31号都是普通的进程信号34~64都是属于实时信号实时信号即是要 立即处理 的信号,而普通信号不需要立即处理。 这些信号在使用的时候,可以直接使用各信号前面的数字,也可以使用信号字母。事实上,这些信号都是宏,这些宏都是定义在  signum.h 的头文件中。


 五 信号的保存

我们知道信号是发给进程的, 而进程要保存信号,那么应该保存在哪里?

我们知道每一个进程的创建,系统都会为它维护一个PCB,里面存储着有关于进程的各种信息,那么是不是有关于进程的信号也应该存储在里面呢,是的。其实,进程信号就是系统通过修改PCB中的数据来打到向进程发送信号的目的。在每个进程的PCB中都描述着一个记录进程信号的 位图 ,当向进程发送信号时,操作系统就会在进程信号位图的指定位置写入1,指定位置的数据变成了1,就说明已经写入了相应的信号,即进程就已经接收到了相应的信号。

发送信号的本质:修改PCB中的信号位图。


六 信号的产生

  • 通过终端按键产生信号 
  • 通过系统调用向进程发送信号
  • 硬件异常产生信号
  • 软件条件产生信号

1. 通过终端按键产生信号 

我们接下来写一个小测试:

 运行结果

✍我们通过 Ctrl +C 使得一个进程退出。 本质上 Ctrl +C是一个组合键----->OS------->OS将

Ctrl +C 解释成2信号相应的进程内核结构位图第二号比特为变为1,但是此时信号并不会立刻处理,我们看到的Ctrl +C 进程就立马退出了,那是我们肉眼所看,实际cpu速度很快,信号并不会立马处理。我们并没有对信号做任何处理,那么2号信号就会进行默认处理。

如何查看信号的默认动作呢? 我们可以通过 man 7  signal 手册进行查看

如何证明  Ctrl +C 就是2号信号呢,我们可以通过系统调用函数 sighal() ,当收到我们指定信号的时候,会调用我们设置的回调函数执行对应的动作。

🍊 系统调用函数 signal()

 

 运行结果

我们要想使得进程退出,可以使用 kill  9 PID

🌾:除了快捷键 ctrl +c 之外。ctrl +\ 也可以使得进程退出 

 

2. 通过系统调用向进程发送信号

㊀: kill

接下来我们写一个测试来进行系统调用向进程发生信号

#include <iostream>
#include <string>
#include <unistd.h>
#include <signal.h>
#include <sys/types.h>
#include <cstdio>
#include <stdlib.h>
//mysignal.cpp
static void Usage(const std::string & proc)
{
    std::cout<< " Usage: "<<proc<<"pid sighno"<<std::endl;
}
// ./myprocess pid signo
int main(int argc, char* argv[])
{
    //2.系统调用向进程发送信号
    if(argc != 3)
    {
        Usage(argv[0]);
        exit(1);
    }
    pid_t pid =atoi(argv[1]);
    int signo =atoi(argv[2]);//将字符串转换成整型

    int n=kill(pid,signo);
    if(n !=0)
    {
        perror("kill");
    }
}
#include <iostream>
#include <sys/types.h>
#include <unistd.h>
//mytest.cpp
int main()
{
    while(true)
    {
        std::cout<<"我是一个正在运行的进程:pid"<<getpid()<<std::endl;
        sleep(1);
    }
}
makefile///
.PHONY:all
all: mysignal mytest

mytest:mytest.cpp
	g++ -o $@ $^ -std=c++11
mysignal:mysignal.cpp
	g++ -o $@ $^ -std=c++11
.PHONY:clean
clean:
	rm -f mysignal mytest

测试结果:

🍎:从这个测试中我们可以认识到,向目标进程发信号,不一定要通过键盘热键,还可以通过系统调用接口。上述测试可以认为我们自主实现了kill 命令。 

㊁:raise() 

上述的kill()接口可以给任意进程发生信号,而raise()接口只能给自己发送信号。 

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

static void Usage(const std::string & proc)
{
    std::cout<< " Usage: "<<proc<<"pid sighno"<<std::endl;
}
// ./myprocess pid signo
int main(int argc, char* argv[])
{
    int cnt=0;
    while(cnt<=10)
    {
        printf("cnt: %d\n",cnt++);
        sleep(1);
        if(cnt>=5)
        {
            raise(3);//向自己发送3号信号结束进程。
        }
    }
 
}

测试结果: 

 ㊂:abort()

abort()给自己发送指定信号(SIGABRT)。

#include <iostream>
#include <string>
#include <unistd.h>
#include <signal.h>
#include <sys/types.h>
#include <cstdio>
#include <stdlib.h>
static void Usage(const std::string & proc)
{
    std::cout<< " Usage: "<<proc<<"pid sighno"<<std::endl;
}
// ./myprocess pid signo
int main(int argc, char* argv[])
{
    int cnt=0;
    while(cnt<=10)
    {
        printf("cnt: %d\n",cnt++);
        sleep(1);
        if(cnt>=5)
        {
            abort();//向自己发送3号信号结束进程。
        }
    }
}

🍏: 关于信号处理行为的理解:进程收到大部分的信号,默认处理动作都是终止进程,既然大多数信号处理动作都是终止进程,为什么还有那么多的信号呢?这是因为信号的不同,代表不同的事件,但是对事件发生之后处理的动作可以一样。

 3. 硬件异常产生信号

信号产生不一定非要用户显示发送,操作系统内部也可以发送信号。

int main(int argc, char* argv[])
{
    std::cout<<"我在运行中..........."<<std::endl;
    sleep(1);
    int a=20;
    a/=0;
}

接下来我们通过 kill -l 来查看对应的信号编号

 再然后我们通过 man 7 signal 手册来查看SIGFPE信号对应的信息,其中Action中有 Core 和

Term ,两者都是终止进程,但是有区别,我们后续再进行说明。

 我们如何证明 Floating point exception 对应的就是8信号呢,接下来我们写个测试,来捕捉除0错误时,系统发来的信号是8信号。

void catchSig(int signo)
{
    std::cout<< "获取到一个信号,信号编号是: "<<signo<<std::endl;
}

int main(int argc, char* argv[])
{
    //如果没有/0错误,这个函数不会进行调用
    signal(SIGFPE,catchSig);
    std::cout<<"我在运行中..........."<<std::endl;
    sleep(1);
    int a=20;
    a/=0;
}

 运行结果:

 🍈:1.OS是如何得知给当前进程发送8号信号的? OS怎么知道我们除0错误了呢?

          2.我们只进行了一次除0,为什么系统一直给我们发送8信号?

1. OS是如何得知给当前进程发送8号信号的? OS怎么知道我们除0错误了呢?

 OS----->cpu运算异常--------->OS必须知道(OS是软硬件资源的管理者)----->OS向进程发送异常信号。

 2.我们只进行了一次除0,为什么系统一直给我们发送8信号?

我们在当前进程进行信号捕捉的时候自定义了信号行为,即进程收到信号并不会退出,既然没有退出,那么进程就还会再被调度。CPU内部只有一份寄存器,CPU再进行运行的时候会进行多个进程切换,也就是说/0这个进程会被调度多次,所以OS就会发送无数次8信号。

4. 软件条件产生信号 

例如管道问题,当管道读端关闭,写端一直写的时候,系统会发送信号终止进程,这个信号是由软件条件(读端关闭,写端一直写)触发的,与硬件无关了,所以叫软件条件产生信号

软件产生的信号通常是指在计算机系统中,操作系统或应用程序通过特定的事件、错误情况或者编程接口(API)调用而触发的一种通知机制。这些信号可以用来通知进程发生了某些特定事件,例如用户请求终止一个进程、硬件异常(如分母为 0 的除法),定时器到期等。

 SIGALRM 是一个软件产生的信号,我们可以在程序内调用 alarm()(alarm闹钟的意思) 接口来设置闹钟。等到“闹钟”响的时候,系统就会向进程发送 SIGALRM 信号(编号14),此信号的默认处理方式为终止进程。

alarm()

 测试:

int main(int argc, char* argv[])
{
    //4.软件条件
    alarm(1);//设置了一秒闹钟,当程序运行1秒之后,程序退出。
    int cnt=0;
    while (true)
    {
        std::cout<< "cnt: "<<cnt++<<std::endl;
    }
}


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

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

相关文章

CSS--解决float: right在空间不够时会自动往下移的问题

原文网址&#xff1a;CSS--解决float: right在空间不够时会自动往下移的问题-CSDN博客 简介 众所周知&#xff0c;float: right在空间不够时会自动往下移。那么怎样让它不要往下移呢&#xff1f;本文介绍解决方案。 需求 我想写一个无需列表&#xff0c;每个列表后边跟一个…

深度学习 Deep Learning 第14章 自编码器

深度学习 Deep Learning 第14章 自编码器 内容概要 本章深入探讨了自编码器&#xff08;Autoencoders&#xff09;&#xff0c;这是一种用于特征学习和降维的神经网络架构。自编码器通过编码器和解码器两个部分&#xff0c;将输入数据映射到一个内部表示&#xff08;编码&…

C++(匿名函数+继承+多态)

#include <iostream> #include <cstring> #include <cstdlib> #include <unistd.h> #include <sstream> #include <vector> #include <memory>using namespace std;// 基类 Weapon class Weapon { protected:int atk; public:Weapon…

软考中级网络工程师第十一章网络管理

11-1考点分析 11-2网络管理基础&#xff08;记忆&#xff09; 网络管理体系结构 网络管理五大功能域&#xff1a;故障管理、配置管理、计费管理、性能管理和安全管理。 助记&#xff1a; “安配能计障” 故障管理&#xff1a;尽快发现故障&#xff0c;找出故障原因&#x…

创维E900V22C/E900V22D_S905L3(B)_安卓9.0_指示灯正常_线刷固件包

创维E900V22C&#xff0f;E900V22D_S905L3(B)_安卓9.0_指示灯正常_线刷固件包 线刷方法&#xff1a;&#xff08;新手参考借鉴一下&#xff09; 1、准备好一根双公头USB线刷刷机线&#xff0c;长度30-50CM长度最佳&#xff0c;同时准备一台电脑&#xff1b; 2、电脑上安装好刷…

“京数青算“启新篇|北方算网与海东市数据局签署合作协议

近日&#xff0c;青海省海东市2025年“京数青算”推介会在北京召开。海东市委常委、副市长梁荣勃&#xff0c;海东市数据局局长安志忠出席会议&#xff0c;北方算网副总经理&#xff08;主持工作&#xff09;喻一鸣等60余家人工智能企业的代表参会。 梁荣勃在致辞中代表海东市…

QML输入控件: Slider的高级外观定制(音视频控制条)

目录 引言相关阅读示例1&#xff1a;基础样式定制要点效果 示例2&#xff1a;音量控制滑块要点效果 示例3&#xff1a;视频进度条要点效果 解决问题总结工程下载 引言 在现代用户界面设计中&#xff0c;滑块控件(Slider)是一个不可或缺的交互元素。它不仅能让用户直观地进行数…

密码学基础——古典密码学

目录 一、定义 特点&#xff1a; 二、发展阶段 三、代换密码 1.单表代换密码 1.1恺撒密码 1.2 移位变换 1.3 仿射变换 2.多表代换密码 维吉尼亚密码 四、置换密码 栅栏密码 一、定义 古典密码学是指在现代密码学出现之前&#xff0c;使用较为简单的数学方法和手工…

KingbaseES物理备份还原之备份还原

此篇续接上一篇<<KingbaseES物理备份还原之物理备份>>,上一篇写物理备份相关操作,此篇写备份还原的具体操作步骤. KingbaseES版本:V009R004C011B003 一.执行最新物理备份还原 --停止数据库服务,并创建物理备份还原测试目录 [V9R4C11B3192-168-198-198 V8]$ sys_ct…

jdk21新特性详解使用总结

jdk21新特性详解总结 1.StringBuilder和StringBuffer新增了一个repeat方法 /*** Java 21的StringBuilder和StringBuffer新增了一个repeat方法*/public static void repeatStr(){var sbnew StringBuilder().repeat("*",10);System.out.println(sb);}运行结果如下&…

【实用技巧】电脑重装后的Office下载和设置

写在前面&#xff1a;本博客仅作记录学习之用&#xff0c;部分图片来自网络&#xff0c;如需引用请注明出处&#xff0c;同时如有侵犯您的权益&#xff0c;请联系删除&#xff01; 文章目录 前言下载设置总结互动致谢参考目录导航 前言 在数字化办公时代&#xff0c;Windows和…

206. 反转链表 92. 反转链表 II 25. K 个一组翻转链表

leetcode Hot 100系列 文章目录 一、翻转链表二、反转链表 II三、K 个一组翻转链表总结 一、翻转链表 建立pre为空&#xff0c;建立cur为head&#xff0c;开始循环&#xff1a;先保存cur的next的值&#xff0c;再将cur的next置为pre&#xff0c;将pre前进到cur的位置&#xf…

离线语音识别 ( 小语种国家都支持)可定制词组

1产品介绍 离线语音模组采用神经网络算法&#xff0c;支持语音识别、自学习等功能。运用此模组将 AI 技 术赋能产品&#xff0c;升级改造出语音操控的智能硬件 ( 例如风扇、台灯、空调、马桶、按摩椅、运 动相机、行车记录仪等 ) 。支持全球多种语言识别&#xff0c;如中文…

网络华为HCIA+HCIP 策略路由,双点双向

目录 路由策略&#xff0c;策略路由 策略路由优势 策略路由分类 接口策略路由 双点双向 双点双向路由引入特点: 联系 路由回灌和环路问题 路由策略&#xff0c;策略路由 路由策略:是对路由条目进行控制&#xff0c;通过控制路由条目影响报文的转发路径&#xff0c;即路…

【TI MSPM0】ADC DAC学习

一、样例展示 通过ADC0触发单次采样&#xff0c;如果采样结果大于0.5倍的VDD&#xff0c;就点亮LED 否则熄灭LED 编译加载运行这个历程&#xff0c;提供一个电压到A0_2引脚上,电压范围在0-VCC之间同时观察LED1.在上电后&#xff0c;默认将ADC配置到正确的引脚模式&#xff0c;…

Cesium系列:从入门到实践,打造属于你的3D地球应用

一、Cesium简介 CesiumJS 是一个开源的 JavaScript 库&#xff0c;它能够帮助开发者创建出具有卓越性能、高精度、出色视觉质量和易用性的世界级 3D 地球仪和地图。无论是在航空航天领域&#xff0c;用于模拟飞行路径和展示卫星数据&#xff1b;还是在智能城市中&#xff0c;用…

Linux系统程序设计:从入门到高级Day01

知识点1 【系统调用】 系统调用的概述 系统调用&#xff1a;内核 提供给 用户 可以 操作内核 的一组函数接口 关系&#xff1a;用户 借助 系统调用 操作内核 进程的空间分为&#xff1a;内核空间 和 用户空间 用户一般都是在用户空间操作的&#xff0c;但是有的时候用户需要…

openEuler24.03 LTS下安装HBase集群

前提条件 安装好Hadoop完全分布式集群&#xff0c;可参考&#xff1a;openEuler24.03 LTS下安装Hadoop3完全分布式 安装好ZooKeeper集群&#xff0c;可参考&#xff1a;openEuler24.03 LTS下安装ZooKeeper集群 HBase集群规划 node2node3node4MasterBackup MasterRegionServ…

关于testng.xml无法找到类的问题

问题&#xff1a;testng.xml添加测试类的时候飘红 解决办法&#xff1a; 1.试图通过自动生成testng.xml插件去解决&#xff0c;感觉也不是这个问题&#xff0c;没有尝试&#xff1b; 2.以为是创建包的方式不对&#xff0c;重新删除后新建--还是找不到 想新建类的时候发现从m…

数据结构:探秘AVL树

本节重点 理解AVL树的概念掌握AVL树正确的插入方法利用_parent指针正确更新平衡因子掌握并理解四种旋转方式&#xff1a;左单旋&#xff0c;右单旋&#xff0c;左右双旋&#xff0c;右左双旋 一、AVL树的概念 AVL树得名于它的发明者G. M. Adelson-Velsky和E. M. Landis&…