Linux--信号--信号的产生方式--核心转储--0104

news2025/1/16 16:35:44

1. 什么是信号

生活中的信号:红绿灯,狼烟,撤退、集合...。

  • 我们认识这些信号,首先是因为自己记住了对应场景下的信号+后续需要执行的动作。
  • 如果信号没有产生,我们依旧知道如何处理这个信号。
  • 收到信号,我们不一定会马上处理。
  • 在我们无法立即处理时,也一定要被先记住。

1.2 Linux信号

本质是一种通知机制,用户或者操作系统通过发送一定的信号,通知进程,某些事情已经发生,需要后续处理。

1.3 kill -l 查看系统定义的信号列表

 没有0、32、33号信号,一共62个

[1,31]普通信号   [34,64]实时信号


2. 信号如何产生

比如:在死循环程序中,通过命令行指令ctrl+c可以终止进程。原因是按ctrl+c键盘会产生一个硬件中断,被OS获取解释成了信号,发送给前台的进程,进程收到信号,进行处理。

信号处理的常见方式:

  • 默认。执行程序员写好的逻辑。
  • 忽略。
  • 自定义。自己写处理逻辑

信号如何被记录?

信号是发给进程的,在进程内需要被记录,也就意味着pcb结构体中有一块内存保存信号的相关数据结构(unsigned int 位图),可以标记普通信号是否被发送。


3.信号的捕捉

3.1 signal函数

 handler是一个回调函数,当进程获得了一个信号,就会去调用这个函数。

#include<iostream>
#include<signal.h>
#include<unistd.h>
using namespace std;

//自定义的信号处理方法
void catchfunc(int signum)
{
    cout<<"进程捕捉到一个信号,正在处理中:"<<signum<<"Pid:"<<getpid()<<endl;
    //捕捉到什么信号 是在signal函数中先设定好的
}
int main()
{
    //signal(2,catchfunc);//当遇到了2号信号 就会自动调用我们自定义的方法
    signal(SIGINT,catchfunc);//SIGINT是宏 也是2号信号 每个信号都有自己的编号和对应的宏
    while(true)
    {
        cout<<"我是一个进程,我正在运行...,pid"<<getpid()<<endl;
        sleep(1);
    }
    return 0;
}

 注意在上面的程序中我们将2号信号的处理方法替换成了打印一句话,于是我们在ctrl+c或者在命名行输入kill -2 pid都无法终止该程序。

 3.2 signal函数的补充说明

  • 特定信号的处理动作,一般只有一个
  • signal函数仅仅是修改进程对特定信号的后续处理动作,而不是直接调用这个后续处理动作。当进程收到这个信号的时候,我们才会进入到这个后续处理动作函数。
  • signal函数一般都写在最前面,类似于先设置一个方法,后续程序出现对应的信号才有我们自定义的方法可以用,相当于配置了环境

4. 信号的产生的方式

4.1通过终端按键产生信号

我们在上述程序中将二号信号的处理方法改了,那我们现在如何关闭进程呢?答案是选择其他信号以终止进程。比如3号信号SIGQUIT。3号信号的默认处理动作是终止进程并且Core Dump,什么是Core Dump呢?

4.1.1 Core Dump 核心转储

当一个进程要异常终止时,可以选择把进程的用户空间内存数据全部保存到磁盘上,文件名通常是core,这叫做Core Dump。

进程异常终止通常是因为有Bug,比如非法内存访问导致段错误,事后可以用调试器检查core文件以查清错误原因,这叫做Post-mortem Debug(事后调试)。

一个进程允许产生多大的core文件取决于进程的Resource Limit(这个信息保存 在PCB中)。默认是不允许产生core文件的,因为core文件中可能包含用户密码等敏感信息,不安全。

查看core文件大小

Bash: ulimit -a

修改core的大小

Bash:ulimit -c 1024

 

 4.1.2 core文件的产生

我们先把对2号信号的自定义处理函数注释掉,分别用2号信号和3号信号终止程序。发现3号命令会在当前路径下生成一个core.pid文件。该文件的编码模式是看不懂的。是用于gdb调试时,找到错误信息的。

 除此之外,当我们在程序中出现了/0错误,进程会收8号信号,然后终止程序并生成core文件。

gdb core-file core.pid

子进程出现/0问题也会出现核心转储

4.2 调用系统函数向进程发送信号

4.2.1 kill 函数

kill命令是调用kill函数实现的。kill函数可以给一个指定的进程发送指定的信号。

#include <signal.h>
int kill(pid_t pid, int signo);

成功返回0 失败返回-1

static void Usage(string proc)
{
    cout<<"Usage:\r\n\t"<<proc<<"signumber processid"<<endl;
}

//需要写 ./mykill 2 pid
int main(int argc,char *argv[])
{
    if(argc!=3)
    {
        Usage(argv[0]);
        exit(1);
    }
    int signumber=atoi(argv[1]);
    int procid=atoi(argv[2]);
    kill(procid,signumber);
    return 0;
}

4.2.2 raise 函数

raise函数可以给当前进程发送指定的信号(自己给自己发信号)。

#include <signal.h>
int raise(int signo);
成功返回0 失败返回-1

int main()
{
    cout<<"我开始运行了"<<endl;
    sleep(1);
    raise(8);//执行到这里就给自己发送8号信号
    return 0;
}

4.2.3 abort 函数

没有参数,会发生转储。给自己发送abort信号(6号信号)

相当于raise(6);

4.2.4 总结

用户调用系统接口发送信号,其实就是OS执行对应的系统调用代码,OS提取参数或给特定的信号编号,然后修改目标进程的信号标记位以完成写信号。进程收到信号,然后执行后续操作。

4.3 由软件异常产生的信号

4.3.1 SIGPIPE

SIGPIPE是一种由软件条件产生的信号。

  • 如果管道写入端对应的文件描述符被关闭,则read会把数据全部读完,最后返回0。
  • 如果管道读入端对应的文件描述符被关闭,则操作系统会终止写入端的程序。

也就是给了SIGPIPE信号。

4.3.2 SIGALRM

SIGALRM是由alarm函数产生的。该信号的默认处理动作就是终止当前程序。

#include <unistd.h>
unsigned int alarm(unsigned int seconds);
调用alarm函数可以设定一个闹钟,也就是告诉内核在seconds秒之后给当前进程发SIGALRM信号。

 验证1s内我们一共会进行多少次count++

int count=0;
void catchfunc()
{
    cout<<"final count"<<count<<endl;
}
int main()
{
    alarm(1);
    signal(SIGALRM,catchfunc);
    while(true) count++;
    return 0;
}

4.4 硬件异常产生信号 

4.4.1 除0错误

void catchfunc(int signumber)
{
    sleep(1);
    cout<<"获得了一个信号"<<signumber<<endl;
}
int main()
{
    signal(SIGFPE,catchfunc);
    int a=100;
    a/=0;
    while(true) sleep(1);
    return 0;
}

 为什么我们这里会一直循环呢?


如何理解除0错误

进行计算的是的CPU这个硬件,CPU内部有寄存器(状态寄存器——位图),除0错误会导致溢出,位图中有对应的状态标记位为溢出标记位,运算完成后OS会自动检测,如果溢出标记位是1。操作系统会识别到有溢出问题。找到当前正在运行的进程的pid,OS发送信号,进程会在合适的时候处理信号。

而寄存器中的数据我们并没有权限去修改,即我们除了终止程序什么也做不了。就会不断的重复发送信号。只有终止程序,上下文数据被释放,寄存器的问题才能被修正。

4.4.2 访问空指针/野指针问题

将上述代码的/0错误换成解引用空指针错误。

信号类型是 SIGSEGV

如何理解 解引用空指针 /野指针/ 指针越界问题

我们首先需要通过地址找到目标的所在位置,而我们的虚拟地址和物理地址之间的转换需要页表和MMU进行处理。MMU是一个硬件,也存在自己的寄存器。当我们的访问非法地址的时候,一定会出现寄存器错误。

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

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

相关文章

springboot学习(七十八) springboot中通过自定义注解实现数据脱敏的功能

文章目录前言一、引入hutools工具类二、定义常用需要脱敏的数据类型的枚举三、定义脱敏方式枚举四、自定义脱敏的注解五、自定义Jackson的序列化方式六、使用七、脱敏效果前言 对于某些接口返回的信息&#xff0c;涉及到敏感数据的必须进行脱敏操作&#xff0c;例如银行卡号、…

带你了解ssh服务过程

远程连接服务 1、什么是远程连接服务器 远程连接服务器通过文字或图形接口方式来远程登录系统&#xff0c;让你在远程终端前登录linux主机以取得可操作主机接口&#xff08;shell&#xff09;&#xff0c;而登录后的操作感觉就像是坐在系统前面一样。 2、远程连接服务器的功…

【C++】函数重载的使用及原理

概述 在学校里&#xff0c;我们都会有班里同学被起外号的经历&#xff0c;而且同一个人可能还会有好几个外号。 在自然语言中&#xff0c;一个词可以有多重含义&#xff0c;人们可以通过上下文来判断该词真实的含义&#xff0c;即该词被重载了。 目录 概述 什么是函数重载 …

项目管理:如何制作项目进度计划表?

项目进度管理是根据项目目标&#xff0c;编制合理的进度计划&#xff0c;并在项目推进过程中随时检查项目执行情况。 项目进度管理的目的就是为了实现最优工期&#xff0c;多快好省地完成任务。 而甘特图&#xff0c;就是用表格图形的方式来展示项目的进展&#xff0c;是一个比…

赛狐ERP:优秀的亚马逊运营具备的五项能力!

我们都知道&#xff0c;亚马逊运营是整个店铺的主导&#xff0c;很大程度上会影响着一个店铺经营的好坏&#xff0c;那么一个好的亚马逊运营&#xff0c;应该具备哪些能力呢&#xff1f;今天赛狐ERP就来给和大家聊一聊&#xff0c;希望对各位亚马逊运营们会有启发&#xff01;1…

ORB-SLAM2 --- LocalMapping::Run 局部建图线程解析

目录 一、线程作用 二、局部建图线程主要流程 三、局部建图线程主函数 四、调用函数解析 4.1 设置"允许接受关键帧"的状态标志LocalMapping::SetAcceptKeyFrames函数解析 4.2 查看列表中是否有等待被插入的关键帧LocalMapping::CheckNewKeyFrames函数 4.3 …

十分钟学会在linux上部署chrony服务器(再见 NTP,是时候拥抱下一代时间同步服务 Chrony 了)

chrony服务器 Chrony 相较于 NTPD 服务的优势 安装与配置&#xff08;Chrony的配置文件是/etc/chrony.conf&#xff09; 同步网络时间服务器 设置开机启动&#xff0c;重启服务 chronyc sources 输出结果解析 练习 实验模型图如下 实验a如下 实验b如下 再见 NTP&#x…

中国手机市场全面衰退,连苹果也未能幸免,大跌近三成

CINNO公布了11月份国内手机市场的数据&#xff0c;数据显示2022年11月份中国市场的手机出货量同比下滑21.7%&#xff0c;在整体大环境出现销量下滑的情况下&#xff0c;此前曾持续逆势增长的苹果也顶不住了&#xff0c;苹果在中国市场的出货量也出现了下滑的势头。数据显示2022…

06-Alibaba Nacos注册中心源码剖析

Nacos&Ribbon&Feign核心微服务架构图 架构原理 1、微服务系统在启动时将自己注册到服务注册中心&#xff0c;同时外发布 Http 接口供其它系统调用(一般都是基于SpringMVC) 2、服务消费者基于 Feign 调用服务提供者对外发布的接口&#xff0c;先对调用的本地接口加上注…

JS继承有哪些,你能否手写其中一两种呢?

引言 JS系列暂定 27 篇&#xff0c;从基础&#xff0c;到原型&#xff0c;到异步&#xff0c;到设计模式&#xff0c;到架构模式等&#xff0c; 本篇是 JS系列中第 3 篇&#xff0c;文章主讲 JS 继承&#xff0c;包括原型链继承、构造函数继承、组合继承、寄生组合继承、原型…

前端vue项目发送请求不携带cookie(vue.config.js和nginx反向代理)

一、本地环境——使用vue.config.js配置了跨域代理本来发现问题&#xff0c;是因为后台记录到接收到的sessionId一直在变化&#xff0c;导致需要在同一个sessionId下处理的逻辑无法实现。一开始以为是前后端分离跨域导致的&#xff0c;网上给出了解决方案&#xff1a;main.js中…

线程同步的实现

线程同步 同步就是协同步调&#xff0c;按预定的先后次序进行运行。如:你说完&#xff0c;我再说。 "同"字从字面上容易理解为一起动作 其实不是&#xff0c;"同"字应是指协同、协助、互相配合。 如进程、线程同步&#xff0c;可理解为进程或线程A和B一…

USB子系统简述

引子&#xff1a;关于 lsusb 命令 lsusb 列出系统中所有的USB设备&#xff1a; Bus 004 Device 001: ID 1d6b:0003 Linux Foundation 3.0 root hubBus 004 &#xff1a;表示第四个 usb 主控制器&#xff08;机器上总共有四个 usb 主控制器&#xff0c;可以通过命令 lspci | g…

看完这篇文章终于弄明白了什么是 RocketMQ 的存储模型

RocketMQ 优异的性能表现&#xff0c;必然绕不开其优秀的存储模型 。这篇文章&#xff0c;笔者按照自己的理解 , 尝试分析 RocketMQ 的存储模型&#xff0c;希望对大家有所启发。1 整体概览首先温习下 RocketMQ 架构。整体架构中包含四种角色 :Producer &#xff1a;消息发布的…

基于Python深度学习的垃圾分类代码,用深度残差网络构建

垃圾分类 完整代码下载地址&#xff1a;基于Python深度学习的垃圾分类代码 介绍 这是一个基于深度学习的垃圾分类小工程&#xff0c;用深度残差网络构建 软件架构 使用深度残差网络resnet50作为基石&#xff0c;在后续添加需要的层以适应不同的分类任务模型的训练需要用生…

Qt扫盲-QSerialPort理论总结

QSerialPort理论总结一、概述二、使用流程1. 错误处理2. 阻塞串行端口编程3. 非阻塞串行端口编程三、信号四、注意事项一、概述 QSerialPort 类其实就是一个打开串口&#xff0c;进行串口通信传输数据的功能类。我们可以使用QSerialPortInfo帮助类获取有关可用串行端口的信息&…

JavaEE高阶---Spring AOP

一&#xff1a;什么是Spring AOP&#xff1f; 首先&#xff0c;AOP是一种思想&#xff0c;它是对某一类事情的集中处理。 如用户登录权限的效验&#xff0c;没学 AOP 之前&#xff0c;我们所有需要判断用户登录的页面&#xff0c;都要各自实现或调用验证的方法。然后有了 AOP …

【Linux进程间通信】

Linux进程间通信进程间通信介绍进程间通信的概念进程间通信的目的进程间通信的本质进程间通信的分类管道什么是管道匿名管道匿名管道的原理pipe函数匿名管道使用步骤匿名管道读写规则匿名管道的特点匿名管道的四种特殊情况匿名管道的大小命名管道命名管道的原理使用命令创建命名…

【浮点数在内存中的存储规则】

我们知道&#xff0c;整型在内存中的存储比较简单&#xff0c;在内存中都是以二进制来存储的。然而&#xff0c;浮点型在内存中的存储较为复杂。下面来详细探讨&#xff1a; 直接举一个例子&#xff1a; int main() { int n 9; float *pFloat (float *)&n; printf("…

工业树莓派解决传统数据设备数据上云问题

一、前言 工业4.0的浪潮下&#xff0c;许多中小型制造业企业渴望通过数字化转型谋求新的发展动力&#xff0c;然而&#xff0c;在转型之路上常常会面临一个问题&#xff1a;传统数据采集设备数量多、种类杂&#xff0c;不支持比较新颖的现场总线协议或者通信技术&#xff0c;最…