【Linux】进程信号(中)(一)

简介: 【Linux】进程信号(中)

1.信号的产生

硬件异常产生信号

a/=0问题

创建mysignal.cc文件

#include<iostream>
using namespace std;
int main()
{
   int a=10;
   a/=0;
   cout<<"div zero"<<endl;
    return 0;
}

45f88b88eb0e473183844b56b13ee251.png

使用make 生成可执行程序时,a/=0会报警


206c090d9a4d4a218f5db2d0402c74ef.png

依旧可以生成可执行程序mysignal,但是运行可执行程序会报错


为什么除0就报错了呢?

当代码除0时,程序运行后就崩溃了,程序运行变为进程,进程运行代码时出现了非法代码,进程退出了


ce906cb9d10e4d178afc548a5d0b01d9.png

将内存中的指令数据load到CPU中

状态寄存器中有比特位表示当前计算的状态

CPU中有的寄存器保存未来的计算结果,用状态寄存器来表示其计算结果的正确或错误

状态寄存器中有一个比特位为0/1,表示本次计算是否有溢出问题

假设本来有32/64位,除0时,导致有更高的进位,计算机识别有溢出了,若溢出,状态寄存器的溢出标记位就会置1


操作系统发现状态寄存器的标记位为1,即识别到硬件异常,从而会立马向目标进程发送信号

而该信号为:Floating point exception 浮点数异常

7edf882b60dc4215b917dc4dd2edc382.png


FPE为结尾的正好为8号信号

除0的本质就是触发硬件(CPU)异常


验证为8号信号


通过设置使进程不退出

把8进程默认方法变为自定义方法


5ed7ec418eff4c249b4254fb8d80b749.png

67cb9c6ea46548d9aa242b0f0fa1062d.png

再次运行可执行程序就会一直循环打印 ,只能通过其他信号终止进程


为什么会一直循环打印?

操作系统发现溢出标志位被置1,硬件发生了异常,传给进程8号信号,但是由于8号信号实现自定义方法,进程并没有退出,而溢出标志位属于进程的上下文,一直作为1存在,操作系统就会一直检测到标志位是1,从而一直给进程发8号信号


野指针问题

e64c3dfb6edc433da17fe73eb6f01b65.png


p作为指针变量,有4/8个字节空间

1是将100作为地址数据写到p变量中

2是 p作为nullptr,*p取的是内存中的0号地址

*p=100,相当于向0号地址处写入100,但是0号地址并没有申请过, 所以就造成了野指针问题


923fb1da7f3e428c8d4644567debb51f.png

运行可执行程序后,发生段错误


为什么越界会使程序崩溃呢?

实际上语言上所呈现的地址为虚拟地址

685ae88979294b108ca02d4c1cc405bc.png

将虚拟地址通过页表映射到物理内存

页表查询kv关系,查表的动作是由MMU硬件(内存管理单元)完成的

将输入数据导入到MMU中,再通过MMU转出

所以从虚拟地址到物理地址,采用软硬件结合的方式


*p=100,并不是进行写入,而是进行虚拟到物理的转换

若没有映射关系存在,MMU硬件会报错

若有映射关系存在,但是没有权限,MMU直接报错

MMU的报错,会使操作系统识别到,操作系统会找到对应的目标进程中的PCB,发送对应的信号,从而终止进程


f54b6dd663414775a9fccfc4c23772d1.png


Segmentation fault对应11号信号


验证为11号信号


37bcfedef24c4df280d2e0effae6a038.png

5ff92242e75746d2b4d1ffd48d4fb2e8.png

执行可执行程序后,会一直无线循环打印

由于MMU硬件报错没有被修复,一直存在,所以每一次进程被调度,操作系统都会识别到异常,向进程发送11号信号

导致一直无线循环打印


核心转储


0cf8f103b93d4451b525f1e6a474b339.png

在众多信号中,存在Core和Term类型,都可以终止进程

两者之间有什么区别呢?

容我慢慢来说


Linux在系统级别提供了一种能力,可以将一个进程异常的时候,

操作系统可以将该进程在异常的时候,核心代码部分进行核心转储

(将内存中进程的相关数据,全部dump到磁盘中)

一般会在当前进程的运行目录下,形成core.pid的二进制文件,如core.pid就被叫做核心转储文件


在云服务器上看不到核心转储文件,因为在云服务器上默认关闭这个功能


输入 ulimit -a 指令

查看当前系统中特定资源对应的上限


c5939b8169cd4b8e9c86aeabd406cab0.png

core file size 代表核心转储,默认大小为0,不允许当前系统在当前目录下形成core文件

设置核心转储大小

3a249394d59c41cf99f220f4c83505e8.png

通过 ulimit -c +大小,如 core file size大小变为10240

Core与Term的区别

96cb28b2aeee408588321974973b538e.png


通过复制SSH渠道,创建终端2


6e15a4ff8d974da78b05bdd4a31fc600.png

2号信号对应Term ,终止进程

d2ffd483d91748b7af688030cba12394.png


在终端1中运行可执行程序,在终端2中发送2号信号干掉进程


11b3d4e17c1a425584ce6f66261eb29a.png


当干掉进程后,并没有发现以pid结尾的文件

说明使用Term类型的信号,干掉进程后,不发生核心转储



8号信号 Core,浮点数异常


873d8801d0864a85b5ebd11d01f98628.png


在终端1中运行可执行程序,在终端2中发送8号信号干掉进程,并出现core dump即核心转储


d8b91c21519345b9bb7c8ace02800850.png


再次使用 ls -l 指令,发现多出来一个 core.2257的文件 即核心转储文件


Term:终止就是终止,没有多余动作

Core:终止,会先进行核心转储,在终止进程



相关文章
|
5月前
|
并行计算 Linux
Linux内核中的线程和进程实现详解
了解进程和线程如何工作,可以帮助我们更好地编写程序,充分利用多核CPU,实现并行计算,提高系统的响应速度和计算效能。记住,适当平衡进程和线程的使用,既要拥有独立空间的'兄弟',也需要在'家庭'中分享和并行的成员。对于这个世界,现在,你应该有一个全新的认识。
241 67
|
6月前
|
安全 Linux
【Linux】阻塞信号|信号原理
本教程从信号的基本概念入手,逐步讲解了阻塞信号的实现方法及其应用场景。通过对这些技术的掌握,您可以更好地控制进程在处理信号时的行为,确保应用程序在复杂的多任务环境中正常运行。
238 84
|
4月前
|
Web App开发 Linux 程序员
获取和理解Linux进程以及其PID的基础知识。
总的来说,理解Linux进程及其PID需要我们明白,进程就如同汽车,负责执行任务,而PID则是独特的车牌号,为我们提供了管理的便利。知道这个,我们就可以更好地理解和操作Linux系统,甚至通过对进程的有效管理,让系统运行得更加顺畅。
119 16
|
4月前
|
Unix Linux
对于Linux的进程概念以及进程状态的理解和解析
现在,我们已经了解了Linux进程的基础知识和进程状态的理解了。这就像我们理解了城市中行人的行走和行为模式!希望这个形象的例子能帮助我们更好地理解这个重要的概念,并在实际应用中发挥作用。
96 20
|
3月前
|
监控 Shell Linux
Linux进程控制(详细讲解)
进程等待是系统通过调用特定的接口(如waitwaitpid)来实现的。来进行对子进程状态检测与回收的功能。
75 0
|
3月前
|
存储 负载均衡 算法
Linux2.6内核进程调度队列
本篇文章是Linux进程系列中的最后一篇文章,本来是想放在上一篇文章的结尾的,但是想了想还是单独写一篇文章吧,虽然说这部分内容是比较难的,所有一般来说是简单的提及带过的,但是为了让大家对进程有更深的理解与认识,还是看了一些别人的文章,然后学习了学习,然后对此做了总结,尽可能详细的介绍明白。最后推荐一篇文章Linux的进程优先级 NI 和 PR - 简书。
105 0
|
3月前
|
存储 Linux Shell
Linux进程概念-详细版(二)
在Linux进程概念-详细版(一)中我们解释了什么是进程,以及进程的各种状态,已经对进程有了一定的认识,那么这篇文章将会继续补全上篇文章剩余没有说到的,进程优先级,环境变量,程序地址空间,进程地址空间,以及调度队列。
67 0
|
3月前
|
Linux 调度 C语言
Linux进程概念-详细版(一)
子进程与父进程代码共享,其子进程直接用父进程的代码,其自己本身无代码,所以子进程无法改动代码,平时所说的修改是修改的数据。为什么要创建子进程:为了让其父子进程执行不同的代码块。子进程的数据相对于父进程是会进行写时拷贝(COW)。
74 0
|
6月前
|
Linux 数据库 Perl
【YashanDB 知识库】如何避免 yasdb 进程被 Linux OOM Killer 杀掉
本文来自YashanDB官网,探讨Linux系统中OOM Killer对数据库服务器的影响及解决方法。当内存接近耗尽时,OOM Killer会杀死占用最多内存的进程,这可能导致数据库主进程被误杀。为避免此问题,可采取两种方法:一是在OS层面关闭OOM Killer,通过修改`/etc/sysctl.conf`文件并重启生效;二是豁免数据库进程,由数据库实例用户借助`sudo`权限调整`oom_score_adj`值。这些措施有助于保护数据库进程免受系统内存管理机制的影响。
|
6月前
|
存储 Linux 调度
【Linux】进程概念和进程状态
本文详细介绍了Linux系统中进程的核心概念与管理机制。从进程的定义出发,阐述了其作为操作系统资源管理的基本单位的重要性,并深入解析了task_struct结构体的内容及其在进程管理中的作用。同时,文章讲解了进程的基本操作(如获取PID、查看进程信息等)、父进程与子进程的关系(重点分析fork函数)、以及进程的三种主要状态(运行、阻塞、挂起)。此外,还探讨了Linux特有的进程状态表示和孤儿进程的处理方式。通过学习这些内容,读者可以更好地理解Linux进程的运行原理并优化系统性能。
224 4